Received: by 2002:ac0:a5b6:0:0:0:0:0 with SMTP id m51-v6csp940090imm; Fri, 1 Jun 2018 12:16:27 -0700 (PDT) X-Google-Smtp-Source: ADUXVKLuqXmtXtvisR76F36grVcxDS25aytZnBBbWyJ8LtqeAM36cDArf2j9rUa4mei2vCJrfeUW X-Received: by 2002:a17:902:6105:: with SMTP id t5-v6mr12239597plj.138.1527880587500; Fri, 01 Jun 2018 12:16:27 -0700 (PDT) ARC-Seal: i=1; a=rsa-sha256; t=1527880587; cv=none; d=google.com; s=arc-20160816; b=vym++ijtIhFoGhx6+vlusaOFTmgtof24Wz6TIp6McDgNmyEcQSA2mII/LJmbC9xq+l 3Ihy6ddBTNNZAkvvbFjlkrrTfaeTaUhX7LGbR9C6jsPL9O8uFOYTD5qcVcVZzza5eovF dPeRQlf5lgdeTraBAUyupg50B9P8bnO/fbRf+D3CRghlUP4JIXnV7yvYpbv1HSFzE+bL lvPwqVVP91w9Oo/hoIEZLJ1RurS3m8JDtV9IoBRg9GVDp8IGPT76g2052MKKkp4aM8Cs 0iVfusLfPTn0tfv3KLgdv9uajr1oGPHFpphb9Y6/y22mbWn7Wwbk3meGP41mUJNYv2ik BAIw== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:user-agent:in-reply-to :content-disposition:mime-version:references:message-id:subject:cc :to:from:date:arc-authentication-results; bh=HUhiZtoIX+Gk1Wjs/Xr0m+/hC5alN7IPTt1q/misI1M=; b=ijvX6S2erEnF1kxH+roz/iWVXHkNRIs3DhjP5iN3ccQROrT01/JI8RchVqu1lmalLA 3HU9OKjy0Eie5JpQci6Lpr4ZMxItOlCMJBy3ylGqq/GIEHeAA8XvpBTgrK+HwnMjTP/n 6Ztm5vibVLf2i15ygs5Js9nK+a7fAOOY8PpHAUhb968YVQYyOTqDC6IND6IWMuTR6rfJ oXyl1sSBmJIY/Lb3s0W1Wnr4FG9cg1BomX/LBRkZIomaeepnH5gmkhsS8xoIUUzXJYu1 HWZctmhGs87HnsxATOEvZQKxLCUMoz6yuYJHsDZgn52S3Bpkp+QNnHgmS3NLE6AEZhVD J0wg== ARC-Authentication-Results: i=1; mx.google.com; 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=fail (p=NONE sp=NONE dis=NONE) header.from=kernel.org Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id p1-v6si40796882plb.204.2018.06.01.12.16.12; Fri, 01 Jun 2018 12:16:27 -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; 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=fail (p=NONE sp=NONE dis=NONE) header.from=kernel.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1753330AbeFATPt (ORCPT + 99 others); Fri, 1 Jun 2018 15:15:49 -0400 Received: from mx2.suse.de ([195.135.220.15]:48129 "EHLO mx2.suse.de" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1751801AbeFATPr (ORCPT ); Fri, 1 Jun 2018 15:15:47 -0400 X-Virus-Scanned: by amavisd-new at test-mx.suse.de Received: from relay1.suse.de (charybdis-ext-too.suse.de [195.135.220.254]) by mx2.suse.de (Postfix) with ESMTP id F33B8AED1; Fri, 1 Jun 2018 19:15:45 +0000 (UTC) Date: Fri, 1 Jun 2018 21:15:45 +0200 From: "Luis R. Rodriguez" To: Mimi Zohar , Matthew Garrett Cc: linux-integrity@vger.kernel.org, linux-security-module@vger.kernel.org, linux-kernel@vger.kernel.org, David Howells , "Luis R . Rodriguez" , Eric Biederman , kexec@lists.infradead.org, Andres Rodriguez , Greg Kroah-Hartman , Ard Biesheuvel , Kees Cook , "Serge E . Hallyn" , Stephen Boyd Subject: Re: [RFC PATCH v4 7/8] ima: based on policy prevent loading firmware (pre-allocated buffer) Message-ID: <20180601191545.GP4511@wotan.suse.de> References: <1527616920-5415-1-git-send-email-zohar@linux.vnet.ibm.com> <1527616920-5415-8-git-send-email-zohar@linux.vnet.ibm.com> MIME-Version: 1.0 Content-Type: text/plain; charset=us-ascii Content-Disposition: inline In-Reply-To: <1527616920-5415-8-git-send-email-zohar@linux.vnet.ibm.com> User-Agent: Mutt/1.6.0 (2016-04-01) Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org On Tue, May 29, 2018 at 02:01:59PM -0400, 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. The QCOM_MDT_LOADER calls dma_alloc_coherent() to > allocate this buffer. According to Documentation/DMA-API.txt, > > Consistent memory is memory for which a write by either the > device or the processor can immediately be read by the processor > or device without having to worry about caching effects. (You > may however need to make sure to flush the processor's write > buffers before telling devices to read that memory.) > > Devices using pre-allocated DMA memory run the risk of the firmware > being accessible by the device prior to the kernel's firmware signature > verification has completed. Indeed. And since its DMA memory we have *no idea* what can happen in terms of consumption of this firmware from hardware, when it would start consuming it in particular. If the device has its own hardware firmware verification mechanism this is completely obscure to us, but it may however suffice certain security policies. The problem here lies in the conflicting security policies of the kernel wanting to not give away firmware until its complete and the current inability to enable us to have platforms suggest they trust hardware won't do something stupid. This becomes an issue since the semantics of the firmware API preallocated buffer do not require currently allow the kernel to inform LSMs of the fact that a buffer is DMA memory or not, and a way for certain platforms then to say that such use is fine for specific devices. Given a pointer can we determine if a piece of memory is DMA or not? Seems hacky to use such inferences if we had them anyway... but worth asking... I would suggest we augment the prealloc buffer firmware API to pass a flags argument which helps describe the preallocated buffer, and for now allow us to enable callers to describe if the buffer is kernel memory or DMA memory, and then have the LSMs decide based on this information as well. The qualcomm driver would change to use the DMA flag, and IMA would in turn deny such uses. Once and if platforms want to trust the DMA flag they can later add infrastructure for specifying this somehow. > Loading firmware already calls the security_kernel_read_file LSM hook. > With an IMA policy requiring signed firmware, this patch prevents > loading firmware into a pre-allocated buffer. > > Signed-off-by: Mimi Zohar > Cc: Luis R. Rodriguez > Cc: David Howells > Cc: Kees Cook > Cc: Serge E. Hallyn > Cc: Stephen Boyd > --- > security/integrity/ima/ima_main.c | 26 +++++++++++++++++--------- > 1 file changed, 17 insertions(+), 9 deletions(-) > > diff --git a/security/integrity/ima/ima_main.c b/security/integrity/ima/ima_main.c > index 4a87f78098c8..3dae605a1604 100644 > --- a/security/integrity/ima/ima_main.c > +++ b/security/integrity/ima/ima_main.c > @@ -419,6 +419,15 @@ void ima_post_path_mknod(struct dentry *dentry) > iint->flags |= IMA_NEW_FILE; > } > > +static int read_idmap[READING_MAX_ID] = { > + [READING_FIRMWARE] = FIRMWARE_CHECK, > + [READING_FIRMWARE_PREALLOC_BUFFER] = FIRMWARE_CHECK, > + [READING_MODULE] = MODULE_CHECK, > + [READING_KEXEC_IMAGE] = KEXEC_KERNEL_CHECK, > + [READING_KEXEC_INITRAMFS] = KEXEC_INITRAMFS_CHECK, > + [READING_POLICY] = POLICY_CHECK > +}; > + > /** > * ima_read_file - pre-measure/appraise hook decision based on policy > * @file: pointer to the file to be measured/appraised/audit > @@ -442,18 +451,17 @@ int ima_read_file(struct file *file, enum kernel_read_file_id read_id) > } > return 0; /* We rely on module signature checking */ > } > + > + if (read_id == READING_FIRMWARE_PREALLOC_BUFFER) { > + if ((ima_appraise & IMA_APPRAISE_FIRMWARE) && > + (ima_appraise & IMA_APPRAISE_ENFORCE)) { > + pr_err("Prevent device from accessing firmware prior to verifying the firmware signature.\n"); > + return -EACCES; > + } > + } > return 0; > } > > -static int read_idmap[READING_MAX_ID] = { > - [READING_FIRMWARE] = FIRMWARE_CHECK, > - [READING_FIRMWARE_PREALLOC_BUFFER] = FIRMWARE_CHECK, > - [READING_MODULE] = MODULE_CHECK, > - [READING_KEXEC_IMAGE] = KEXEC_KERNEL_CHECK, > - [READING_KEXEC_INITRAMFS] = KEXEC_INITRAMFS_CHECK, > - [READING_POLICY] = POLICY_CHECK > -}; > - > /** > * ima_post_read_file - in memory collect/appraise/audit measurement > * @file: pointer to the file to be measured/appraised/audit > -- > 2.7.5 > > -- Do not panic