Received: by 2002:a25:8b91:0:0:0:0:0 with SMTP id j17csp3473843ybl; Mon, 27 Jan 2020 04:50:05 -0800 (PST) X-Google-Smtp-Source: APXvYqwhWcd4QrjyZpv+VHvzFHFGUEtHaNlIUzIQ1sVoEzzPvqGqgXBQjP7MS6WyjEIz7HD98e5N X-Received: by 2002:a05:6830:1e72:: with SMTP id m18mr5328602otr.226.1580129405323; Mon, 27 Jan 2020 04:50:05 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1580129405; cv=none; d=google.com; s=arc-20160816; b=jVA543cMEgsE6ROeoKy5gkDKoWnBJmJIIHdbKcwzMqEh/WZV3pEd0zsFpyjrNNyf3m BdHA7RUWA734120jNLci+0iekzIDUgtHHg4TJ9vbayTW7bejZv0b9NcqjX99iMuZCMOV ZXLLCj3BEsRHLlbDNHafCoXv5CK9glXHs/EHiNnhamltScvdvVYa/D1YJSEkCDtwxxpk ZRBnJT1P4usLq8W+dkvQv0ma7xBkhULm4wRCiQbeRFi8KGaNPbA92GWrX4SOugZKviH1 9SNmlvzn1aP1z34JB5JbeY1d1gly8j8bmNnpapqKZh6p9eXOqWllPOQ+MkxKdnu8IqfF nf6w== ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=google.com; s=arc-20160816; h=list-id:precedence:sender:content-transfer-encoding:mime-version :message-id:date:subject:cc:to:from; bh=h8oU2DtLjVrKvNHU1U2WmoJr5hN8FomJD1A+QHBNzg8=; b=vju05rJSylhgielpoNQZGXaqfxk8i0klF2sjLD/u3mLONrL+OmRY/f1tCquDicU2tv E2Us7S90lpIsIjrV3lJwYZdFlyBGEIvms8SFht3fV12x9whWqIC1+46Tj5zXWBtE1A8h N7HiVjo8KWSiMTCJT0GsQiiZmJrf5OR2e6iU7o/+IyHfRCplXmSBbUaqDXldWjKSM9Gk 6s/LfX+rZI2tyQAzAjmqv+wDSXM6tvtFFzfxMTY6pTXS6BEsBuamcbtVnIAkirsTBP2M tJkkhb/rjrI466PD3EmPIOgwVXpZK7zA18F5563tOGtq0sa2t7ZyvDvUwGdAsnmAtiII wu1A== ARC-Authentication-Results: i=1; mx.google.com; spf=pass (google.com: best guess record for domain of linux-crypto-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-crypto-owner@vger.kernel.org Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id t1si5771843otq.148.2020.01.27.04.49.54; Mon, 27 Jan 2020 04:50:05 -0800 (PST) Received-SPF: pass (google.com: best guess record for domain of linux-crypto-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-crypto-owner@vger.kernel.org designates 209.132.180.67 as permitted sender) smtp.mailfrom=linux-crypto-owner@vger.kernel.org Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1729887AbgA0M3s (ORCPT + 99 others); Mon, 27 Jan 2020 07:29:48 -0500 Received: from foss.arm.com ([217.140.110.172]:43792 "EHLO foss.arm.com" rhost-flags-OK-OK-OK-OK) by vger.kernel.org with ESMTP id S1730196AbgA0M3r (ORCPT ); Mon, 27 Jan 2020 07:29:47 -0500 Received: from usa-sjc-imap-foss1.foss.arm.com (unknown [10.121.207.14]) by usa-sjc-mx-foss1.foss.arm.com (Postfix) with ESMTP id 281DF30E; Mon, 27 Jan 2020 04:29:47 -0800 (PST) Received: from e110176-lin.kfn.arm.com (e110176-lin.kfn.arm.com [10.50.4.146]) by usa-sjc-imap-foss1.foss.arm.com (Postfix) with ESMTPSA id C156A3F52E; Mon, 27 Jan 2020 04:29:45 -0800 (PST) From: Gilad Ben-Yossef To: Herbert Xu , "David S. Miller" Cc: Ofir Drang , Geert Uytterhoeven , linux-crypto@vger.kernel.org, linux-kernel@vger.kernel.org Subject: [RFC] crypto: ccree - protect against short scatterlists Date: Mon, 27 Jan 2020 14:29:39 +0200 Message-Id: <20200127122939.6952-1-gilad@benyossef.com> X-Mailer: git-send-email 2.23.0 MIME-Version: 1.0 Content-Transfer-Encoding: 8bit Sender: linux-crypto-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-crypto@vger.kernel.org Deal gracefully with the event of being handed a scatterlist which is shorter than expected. This mitigates a crash in some cases of Crypto API calls due with scatterlists with a NULL first buffer, despite the aead.h forbidding doing so. Signed-off-by: Gilad Ben-Yossef Reported-by: Geert Uytterhoeven --- drivers/crypto/ccree/cc_buffer_mgr.c | 54 ++++++++++++++-------------- 1 file changed, 26 insertions(+), 28 deletions(-) diff --git a/drivers/crypto/ccree/cc_buffer_mgr.c b/drivers/crypto/ccree/cc_buffer_mgr.c index a72586eccd81..62a0dfb0b0b6 100644 --- a/drivers/crypto/ccree/cc_buffer_mgr.c +++ b/drivers/crypto/ccree/cc_buffer_mgr.c @@ -87,6 +87,11 @@ static unsigned int cc_get_sgl_nents(struct device *dev, { unsigned int nents = 0; + *lbytes = 0; + + if (!sg_list || !sg_list->length) + goto out; + while (nbytes && sg_list) { nents++; /* get the number of bytes in the last entry */ @@ -95,6 +100,8 @@ static unsigned int cc_get_sgl_nents(struct device *dev, nbytes : sg_list->length; sg_list = sg_next(sg_list); } + +out: dev_dbg(dev, "nents %d last bytes %d\n", nents, *lbytes); return nents; } @@ -290,37 +297,28 @@ static int cc_map_sg(struct device *dev, struct scatterlist *sg, unsigned int nbytes, int direction, u32 *nents, u32 max_sg_nents, u32 *lbytes, u32 *mapped_nents) { - if (sg_is_last(sg)) { - /* One entry only case -set to DLLI */ - if (dma_map_sg(dev, sg, 1, direction) != 1) { - dev_err(dev, "dma_map_sg() single buffer failed\n"); - return -ENOMEM; - } - dev_dbg(dev, "Mapped sg: dma_address=%pad page=%p addr=%pK offset=%u length=%u\n", - &sg_dma_address(sg), sg_page(sg), sg_virt(sg), - sg->offset, sg->length); - *lbytes = nbytes; - *nents = 1; - *mapped_nents = 1; - } else { /*sg_is_last*/ - *nents = cc_get_sgl_nents(dev, sg, nbytes, lbytes); - if (*nents > max_sg_nents) { - *nents = 0; - dev_err(dev, "Too many fragments. current %d max %d\n", - *nents, max_sg_nents); - return -ENOMEM; - } - /* In case of mmu the number of mapped nents might - * be changed from the original sgl nents - */ - *mapped_nents = dma_map_sg(dev, sg, *nents, direction); - if (*mapped_nents == 0) { + int ret = 0; + + *nents = cc_get_sgl_nents(dev, sg, nbytes, lbytes); + if (*nents > max_sg_nents) { + *nents = 0; + dev_err(dev, "Too many fragments. current %d max %d\n", + *nents, max_sg_nents); + return -ENOMEM; + } + + if (nents) { + + ret = dma_map_sg(dev, sg, *nents, direction); + if (dma_mapping_error(dev, ret)) { *nents = 0; - dev_err(dev, "dma_map_sg() sg buffer failed\n"); + dev_err(dev, "dma_map_sg() sg buffer failed %d\n", ret); return -ENOMEM; } } + *mapped_nents = ret; + return 0; } @@ -881,7 +879,7 @@ static int cc_aead_chain_data(struct cc_drvdata *drvdata, &src_last_bytes); sg_index = areq_ctx->src_sgl->length; //check where the data starts - while (sg_index <= size_to_skip) { + while (src_mapped_nents && (sg_index <= size_to_skip)) { src_mapped_nents--; offset -= areq_ctx->src_sgl->length; sgl = sg_next(areq_ctx->src_sgl); @@ -921,7 +919,7 @@ static int cc_aead_chain_data(struct cc_drvdata *drvdata, offset = size_to_skip; //check where the data starts - while (sg_index <= size_to_skip) { + while (dst_mapped_nents && sg_index <= size_to_skip) { dst_mapped_nents--; offset -= areq_ctx->dst_sgl->length; sgl = sg_next(areq_ctx->dst_sgl); -- 2.23.0