Received: by 2002:ac0:a5a7:0:0:0:0:0 with SMTP id m36-v6csp1357174imm; Mon, 9 Jul 2018 23:52:53 -0700 (PDT) X-Google-Smtp-Source: AAOMgpfgr3lHWiToY6HqcKEI5Yw+tMqo3UvkicC4KExRKV5o+GA63EDXT6Hzv8LFIzgLnd9XmUsg X-Received: by 2002:a63:8341:: with SMTP id h62-v6mr20991487pge.298.1531205573497; Mon, 09 Jul 2018 23:52:53 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1531205573; cv=none; d=google.com; s=arc-20160816; b=zQKLWz+n2K82oSkmflhuqHLcYPgx0elfDFlSlAjzJ0wMyJ/kZxHNwl5msrKv2AMHj0 ci3UAaOPj0HVqXdlmm4CsBCs2Jb8CBdzFlTGf8wsC4SqSmvDPJrF+TUGd5qRx6CEfg7l KWLhUQ2m+RKpW9PSA2AxyobStX+NgLEED+/PRE8uKvjMp0ZZhkRxaNJgzVGIge6ZqAfy edsWekZ0xM2CNpA6vV0dl4YREzqcJUPg71fN0KIJaiN7YBHbUJWAJeUCzWyrbG586RVV B7bIR+TIzHBVSZ7k9+O1OhZaOS905HRZi5AfT1646HpHL3vEZkXy0n11QdQAGs6pOdh/ 1H6Q== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:cc:to:subject:message-id:date:from :references:in-reply-to:mime-version:dkim-signature :arc-authentication-results; bh=2MGVBHamptqp7EKDlxfRmKLs8qTrufU4cPeUT5Ewwag=; b=p505f748cItGxEDt/DYO6Px3bcEB1hdKw4xRdPwexlZmRJfutQBVEJ4jsnb0ohMtK1 Dj13jCEdhX7/RLkxS2ycyika+/BLVNdKNwoaW61NNg/bwpq8sSrTrX+wfcdLTVfKIclw +aCXbvlgmfiHDlKc7rNWPaHJwtfx8mA55WGfKAP6Jp6ntGbgWLYAUqlQ4qWSHm6nu5dW 6NPOYhxYLfiY7UQFcDHBUVZPFxRuvp9S9vWlx1Uy4ufGRp2OCG6thi7XQiDH2OG4gTr4 91C/emSapP8Q0EBsjGacjBHE83RlSD+D+fyBzShp+d2RQUmpEsjbDuzEqYIiiQX3Nqk9 jSBQ== ARC-Authentication-Results: i=1; mx.google.com; dkim=pass header.i=@linaro.org header.s=google header.b=H6I1jeVg; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=linaro.org Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id 37-v6si9168817pgy.118.2018.07.09.23.52.38; Mon, 09 Jul 2018 23:52:53 -0700 (PDT) Received-SPF: pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) client-ip=209.132.180.67; Authentication-Results: mx.google.com; dkim=pass header.i=@linaro.org header.s=google header.b=H6I1jeVg; spf=pass (google.com: best guess record for domain of linux-kernel-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-kernel-owner@vger.kernel.org; dmarc=pass (p=NONE sp=NONE dis=NONE) header.from=linaro.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S932172AbeGJGvp (ORCPT + 99 others); Tue, 10 Jul 2018 02:51:45 -0400 Received: from mail-it0-f67.google.com ([209.85.214.67]:53179 "EHLO mail-it0-f67.google.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1753072AbeGJGvb (ORCPT ); Tue, 10 Jul 2018 02:51:31 -0400 Received: by mail-it0-f67.google.com with SMTP id p4-v6so28969681itf.2 for ; Mon, 09 Jul 2018 23:51:31 -0700 (PDT) DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=linaro.org; s=google; h=mime-version:in-reply-to:references:from:date:message-id:subject:to :cc; bh=2MGVBHamptqp7EKDlxfRmKLs8qTrufU4cPeUT5Ewwag=; b=H6I1jeVgMFJx7oKUzbo4/aqLnPhVU3Ic2Syz0qaUipZev8eFVTFiUpd+0Fc1MvrSNZ RA7u6UzZUDeD2QGO9s5Zq1Mquau/iA/KgN+e38TCZ6ecSnzxZgNjmRMudtFVAkRo9B7m 5/3ho5uQKRnVnkZEVRT6ij0xBNH6VX1B17WEw= X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20161025; h=x-gm-message-state:mime-version:in-reply-to:references:from:date :message-id:subject:to:cc; bh=2MGVBHamptqp7EKDlxfRmKLs8qTrufU4cPeUT5Ewwag=; b=tfwJf1UWchv9lPb7svL2celzthaDE7AVIMlZSj6u5XdCtDSjU9Ue+ww450Djd+3JRe ZYspRfeWuYfsEEdZcDyiaTjo1pbVgjlN47Vy7iVqePOfT0y+YsOnH+2bnoL6BK+rH17m FH5oVSc90JqWKA+WMXnkSJonqFLjyDojQ40ijJ+JMzNOvirbGWUewOhc6f+RV3YrqRxd B8zfageZ58d4EommCNicIC2vOwABp6+/zv0yGufZkDC0z5OrXHLOGN3HKiXnTaBk+a/m v5oldxHrgBegms2f71yAvGzh4XOwajoF/DKu4K6TNCD5QSuhwyNGUZvJE4yYfsS8YqV2 ZG1A== X-Gm-Message-State: APt69E1WH91X6iWuj2EALAiaeUSXGSFYKg++CYJJSAZRH+46DNfdLkrN xLXZh7R8ReugGsaxuvY8a/f3BYThQs1ASRc5So7lXA== X-Received: by 2002:a24:64d6:: with SMTP id t205-v6mr18519866itc.138.1531205490939; Mon, 09 Jul 2018 23:51:30 -0700 (PDT) MIME-Version: 1.0 Received: by 2002:a6b:bbc7:0:0:0:0:0 with HTTP; Mon, 9 Jul 2018 23:51:30 -0700 (PDT) In-Reply-To: <1531165294.3332.40.camel@linux.ibm.com> References: <1530542283-26145-1-git-send-email-zohar@linux.vnet.ibm.com> <1530542283-26145-8-git-send-email-zohar@linux.vnet.ibm.com> <1531165294.3332.40.camel@linux.ibm.com> From: Ard Biesheuvel Date: Tue, 10 Jul 2018 08:51:30 +0200 Message-ID: Subject: Re: [PATCH v5 7/8] ima: based on policy warn about loading firmware (pre-allocated buffer) To: Mimi Zohar Cc: Mimi Zohar , linux-integrity , linux-security-module , Linux Kernel Mailing List , David Howells , "Luis R . Rodriguez" , Eric Biederman , Kexec Mailing List , Andres Rodriguez , Greg Kroah-Hartman , "Luis R . Rodriguez" , Kees Cook , "Serge E . Hallyn" , Stephen Boyd , Bjorn Andersson Content-Type: text/plain; charset="UTF-8" Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org On 9 July 2018 at 21:41, Mimi Zohar wrote: > On Mon, 2018-07-02 at 17:30 +0200, Ard Biesheuvel wrote: >> On 2 July 2018 at 16:38, Mimi Zohar wrote: >> > Some systems are memory constrained but they need to load very large >> > firmwares. The firmware subsystem allows drivers to request this >> > firmware be loaded from the filesystem, but this requires that the >> > entire firmware be loaded into kernel memory first before it's provided >> > to the driver. This can lead to a situation where we map the firmware >> > twice, once to load the firmware into kernel memory and once to copy the >> > firmware into the final resting place. >> > >> > To resolve this problem, commit a098ecd2fa7d ("firmware: support loading >> > into a pre-allocated buffer") introduced request_firmware_into_buf() API >> > that allows drivers to request firmware be loaded directly into a >> > pre-allocated buffer. (Based on the mailing list discussions, calling >> > dma_alloc_coherent() is unnecessary and confusing.) >> > >> > (Very broken/buggy) devices using pre-allocated memory run the risk of >> > the firmware being accessible to the device prior to the completion of >> > IMA's signature verification. For the time being, this patch emits a >> > warning, but does not prevent the loading of the firmware. >> > >> >> As I attempted to explain in the exchange with Luis, this has nothing >> to do with broken or buggy devices, but is simply the reality we have >> to deal with on platforms that lack IOMMUs. > >> Even if you load into one buffer, carry out the signature verification >> and *only then* copy it to another buffer, a bus master could >> potentially read it from the first buffer as well. Mapping for DMA >> does *not* mean 'making the memory readable by the device' unless >> IOMMUs are being used. Otherwise, a bus master can read it from the >> first buffer, or even patch the code that performs the security check >> in the first place. For such platforms, copying the data around to >> prevent the device from reading it is simply pointless, as well as any >> other mitigation in software to protect yourself from misbehaving bus >> masters. > > Thank you for taking the time to explain this again. > >> So issuing a warning in this particular case is rather arbitrary. On >> these platforms, all bus masters can read (and modify) all of your >> memory all of the time, and as long as the firmware loader code takes >> care not to provide the DMA address to the device until after the >> verification is complete, it really has done all it reasonably can in >> the environment that it is expected to operate in. > > So for the non-IOMMU system case, differentiating between pre- > allocated buffers vs. using two buffers doesn't make sense. > >> >> (The use of dma_alloc_coherent() is a bit of a red herring here, as it >> incorporates the DMA map operation. However, DMA map is a no-op on >> systems with cache coherent 1:1 DMA [iow, all PCs and most arm64 >> platforms unless they have IOMMUs], and so there is not much >> difference between memory allocated with kmalloc() or with >> dma_alloc_coherent() in terms of whether the device can access it >> freely) > > What about systems with an IOMMU? > On systems with an IOMMU, performing the DMA map will create an entry in the IOMMU page tables for the physical region associated with the buffer, making the region accessible to the device. For platforms in this category, using dma_alloc_coherent() for allocating a buffer to pass firmware to the device does open a window where the device could theoretically access this data while the validation is still in progress. Note that the device still needs to be informed about the address of the buffer: just calling dma_alloc_coherent() will not allow the device to find the firmware image in its memory space, and arbitrary DMA accesses performed by the device will trigger faults that are reported to the OS. So the window between DMA map (or dma_alloc_coherent()) and the device specific command to pass the DMA buffer address to the device is not inherently unsafe IMO, but I do understand the need to cover this scenario. As I pointed out before, using coherent DMA buffers to perform streaming DMA is generally a bad idea, since they may be allocated from a finite pool, and may use uncached mappings, making the access slower than necessary (while streaming DMA can use any kmalloc'ed buffer and will just flush the contents of the caches to main memory when the DMA map is performed). So to summarize again: in my opinion, using a single buffer is not a problem as long as the validation completes before the DMA map is performed. This will provide the expected guarantees on systems with IOMMUs, and will not complicate matters on systems where there is no point in obsessing about this anyway given that devices can access all of memory whenever they want to. As for the Qualcomm case: dma_alloc_coherent() is not needed here but simply ends up being used because it was already wired up in the qualcomm specific secure world API, which amounts to doing syscalls into a higher privilege level than the one the kernel itself runs at. So again, reasoning about whether the secure world will look at your data before you checked the sig is rather pointless, and adding special cases to the IMA api to cater for this use case seems like a waste of engineering and review effort to me. If we have to do something to tie up this loose end, let's try switching it to the streaming DMA api instead. -- Ard.