Received: by 2002:a25:8b91:0:0:0:0:0 with SMTP id j17csp1543858ybl; Mon, 2 Dec 2019 05:04:56 -0800 (PST) X-Google-Smtp-Source: APXvYqw83prAqMNIeG6qqF8Ptev0qSYofEPioDgr1AjRP5dzQ57rfzDN01uC8qI91ghlfwiK5qTj X-Received: by 2002:a50:fb8f:: with SMTP id e15mr33128770edq.239.1575291896590; Mon, 02 Dec 2019 05:04:56 -0800 (PST) ARC-Seal: i=1; a=rsa-sha256; t=1575291896; cv=none; d=google.com; s=arc-20160816; b=rxEX0rXuaKIAhUVuHkpt83uxYtyal/7XARWaBDdPNL7yfcyxIhuEo9kzofNGeugjSH bCcF9JIHZmAgA3PMDk4lTHzT00xjZB1SEtAEpnsbfXtkjvzIOZNG/Dj/gieuyS8bSRxC z2GAcu5A95FczhFKo2eEdA9NaaXTEsqBKB+r0KC486Eu8tt5HPB12ooeIB9YfrlkQOAv XPOMt8SXFmCt4gmvtp6ReDuAt3JKGNn6Ad300EB+DEt4WLaZKLyhx8SHcEz4e52XMXzY qSx6kk1PQBIJ+XwWgNARZCg802WX1eJUg9H/G6jIwm0uszlbZqqLXsUtfYN4xnR6KtUF i+jg== 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 :references:in-reply-to:message-id:date:subject:cc:to:from; bh=5u638KnIcUYXswIqtr++YRnv15ZZE92jT/ElOuSkL4E=; b=rXxuxElig4AFxJGxIQmLyyw8Zo2pEyV3eXCAGidSRYe24WnjtdFvCe38zbbQ+rVKfQ mCWw+jcmUz6q9P+bjr5YKYwpNLRpRpm9hOR775O6cLAX0ZKBEDa8Ff35oQnrDBBcCrv8 I9AB4k5dnVbETh9JpDRr8s7208R2QZ0ynLe4qavHDZiMwFRXHQdLdL9AnMGSDqbqdrm7 hmmL5O9tZT7zrr4YqHV2GP5IAtpQ83fkHXbSeqxvh5U3Hc41RNPux01M50mOofDhT+4a 1g7Yj3MHc8ZBR1A5MmqpMItcy+3C+7dmzIfGx0IgFcEu1QbO8yPVkrQtkcVua5lKaeMX 1k8Q== 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=QUARANTINE dis=NONE) header.from=gmail.com Return-Path: Received: from vger.kernel.org (vger.kernel.org. [209.132.180.67]) by mx.google.com with ESMTP id k8si2589218ejb.13.2019.12.02.05.04.32; Mon, 02 Dec 2019 05:04:56 -0800 (PST) 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=QUARANTINE dis=NONE) header.from=gmail.com Received: (majordomo@vger.kernel.org) by vger.kernel.org via listexpand id S1727538AbfLBNCt (ORCPT + 99 others); Mon, 2 Dec 2019 08:02:49 -0500 Received: from 212.199.177.27.static.012.net.il ([212.199.177.27]:45795 "EHLO herzl.nuvoton.co.il" rhost-flags-OK-OK-OK-FAIL) by vger.kernel.org with ESMTP id S1727462AbfLBNCk (ORCPT ); Mon, 2 Dec 2019 08:02:40 -0500 Received: from taln60.nuvoton.co.il (ntil-fw [212.199.177.25]) by herzl.nuvoton.co.il (8.13.8/8.13.8) with ESMTP id xB2D1pmQ015197; Mon, 2 Dec 2019 15:01:51 +0200 Received: by taln60.nuvoton.co.il (Postfix, from userid 10140) id 7317260275; Mon, 2 Dec 2019 15:01:51 +0200 (IST) From: amirmizi6@gmail.com To: Eyal.Cohen@nuvoton.com, jarkko.sakkinen@linux.intel.com, oshrialkoby85@gmail.com, alexander.steffen@infineon.com, robh+dt@kernel.org, mark.rutland@arm.com, peterhuewe@gmx.de, jgg@ziepe.ca, arnd@arndb.de, gregkh@linuxfoundation.org Cc: devicetree@vger.kernel.org, linux-kernel@vger.kernel.org, linux-integrity@vger.kernel.org, oshri.alkoby@nuvoton.com, tmaimon77@gmail.com, gcwilson@us.ibm.com, kgoldman@us.ibm.com, ayna@linux.vnet.ibm.com, Dan.Morav@nuvoton.com, oren.tanami@nuvoton.com, shmulik.hager@nuvoton.com, amir.mizinski@nuvoton.com, Amir Mizinski Subject: [PATCH v1 2/5] char: tpm: Add check_data handle to tpm_tis_phy_ops in order to check data integrity Date: Mon, 2 Dec 2019 15:01:29 +0200 Message-Id: <20191202130132.175382-3-amirmizi6@gmail.com> X-Mailer: git-send-email 2.22.0 In-Reply-To: <20191202130132.175382-1-amirmizi6@gmail.com> References: <20191202130132.175382-1-amirmizi6@gmail.com> MIME-Version: 1.0 Content-Transfer-Encoding: 8bit Sender: linux-kernel-owner@vger.kernel.org Precedence: bulk List-ID: X-Mailing-List: linux-kernel@vger.kernel.org From: Amir Mizinski The current principles: - When sending command: 1. Host writes TPM_STS.commandReady 2. Host writes command 3. Host checks TPM received data correctly 4. if not go to step 1 - When receiving data: 1. Host check TPM_STS.dataAvail is set 2. Host get data 3. Host check received data are correct. 4. if not Host write TPM_STS.responseRetry and go to step 1. this commit is based on previous work by Christophe Richard Signed-off-by: Amir Mizinski --- drivers/char/tpm/tpm_tis_core.c | 97 +++++++++++++++++++++++++---------------- drivers/char/tpm/tpm_tis_core.h | 3 ++ 2 files changed, 62 insertions(+), 38 deletions(-) diff --git a/drivers/char/tpm/tpm_tis_core.c b/drivers/char/tpm/tpm_tis_core.c index c3181ea..ce7f8a1 100644 --- a/drivers/char/tpm/tpm_tis_core.c +++ b/drivers/char/tpm/tpm_tis_core.c @@ -242,6 +242,15 @@ static u8 tpm_tis_status(struct tpm_chip *chip) return status; } +static bool tpm_tis_check_data(struct tpm_chip *chip, const u8 *buf, size_t len) +{ + struct tpm_tis_data *priv = dev_get_drvdata(&chip->dev); + + if (priv->phy_ops->check_data) + return priv->phy_ops->check_data(priv, buf, len); + return true; +} + static void tpm_tis_ready(struct tpm_chip *chip) { struct tpm_tis_data *priv = dev_get_drvdata(&chip->dev); @@ -308,47 +317,55 @@ static int tpm_tis_recv(struct tpm_chip *chip, u8 *buf, size_t count) { struct tpm_tis_data *priv = dev_get_drvdata(&chip->dev); int size = 0; - int status; + int status, i; u32 expected; + bool check_data = false; - if (count < TPM_HEADER_SIZE) { - size = -EIO; - goto out; - } + for (i = 0; i < TPM_RETRY && !check_data; i++) { + if (count < TPM_HEADER_SIZE) { + size = -EIO; + goto out; + } - size = recv_data(chip, buf, TPM_HEADER_SIZE); - /* read first 10 bytes, including tag, paramsize, and result */ - if (size < TPM_HEADER_SIZE) { - dev_err(&chip->dev, "Unable to read header\n"); - goto out; - } + size = recv_data(chip, buf, TPM_HEADER_SIZE); + /* read first 10 bytes, including tag, paramsize, and result */ + if (size < TPM_HEADER_SIZE) { + dev_err(&chip->dev, "Unable to read header\n"); + goto out; + } - expected = be32_to_cpu(*(__be32 *) (buf + 2)); - if (expected > count || expected < TPM_HEADER_SIZE) { - size = -EIO; - goto out; - } + expected = be32_to_cpu(*(__be32 *) (buf + 2)); + if (expected > count || expected < TPM_HEADER_SIZE) { + size = -EIO; + goto out; + } - size += recv_data(chip, &buf[TPM_HEADER_SIZE], - expected - TPM_HEADER_SIZE); - if (size < expected) { - dev_err(&chip->dev, "Unable to read remainder of result\n"); - size = -ETIME; - goto out; - } + size += recv_data(chip, &buf[TPM_HEADER_SIZE], + expected - TPM_HEADER_SIZE); + if (size < expected) { + dev_err(&chip->dev, "Unable to read remainder of result\n"); + size = -ETIME; + goto out; + } - if (wait_for_tpm_stat(chip, TPM_STS_VALID, chip->timeout_c, - &priv->int_queue, false) < 0) { - size = -ETIME; - goto out; - } - status = tpm_tis_status(chip); - if (status & TPM_STS_DATA_AVAIL) { /* retry? */ - dev_err(&chip->dev, "Error left over data\n"); - size = -EIO; - goto out; - } + if (wait_for_tpm_stat(chip, TPM_STS_VALID, chip->timeout_c, + &priv->int_queue, false) < 0) { + size = -ETIME; + goto out; + } + + status = tpm_tis_status(chip); + if (status & TPM_STS_DATA_AVAIL) { /* retry? */ + dev_err(&chip->dev, "Error left over data\n"); + size = -EIO; + goto out; + } + check_data = tpm_tis_check_data(chip, buf, size); + if (!check_data) + tpm_tis_write8(priv, TPM_STS(priv->locality), + TPM_STS_RESPONSE_RETRY); + } out: tpm_tis_ready(chip); return size; @@ -453,13 +470,17 @@ static void disable_interrupts(struct tpm_chip *chip) static int tpm_tis_send_main(struct tpm_chip *chip, const u8 *buf, size_t len) { struct tpm_tis_data *priv = dev_get_drvdata(&chip->dev); - int rc; + int rc, i; u32 ordinal; unsigned long dur; + bool data_valid = false; - rc = tpm_tis_send_data(chip, buf, len); - if (rc < 0) - return rc; + for (i = 0; i < TPM_RETRY && !data_valid; i++) { + rc = tpm_tis_send_data(chip, buf, len); + if (rc < 0) + return rc; + data_valid = tpm_tis_check_data(chip, buf, len); + } /* go and do it */ rc = tpm_tis_write8(priv, TPM_STS(priv->locality), TPM_STS_GO); diff --git a/drivers/char/tpm/tpm_tis_core.h b/drivers/char/tpm/tpm_tis_core.h index d06c65b..486c2e9 100644 --- a/drivers/char/tpm/tpm_tis_core.h +++ b/drivers/char/tpm/tpm_tis_core.h @@ -34,6 +34,7 @@ enum tis_status { TPM_STS_GO = 0x20, TPM_STS_DATA_AVAIL = 0x10, TPM_STS_DATA_EXPECT = 0x08, + TPM_STS_RESPONSE_RETRY = 0x02, }; enum tis_int_flags { @@ -106,6 +107,8 @@ struct tpm_tis_phy_ops { int (*read16)(struct tpm_tis_data *data, u32 addr, u16 *result); int (*read32)(struct tpm_tis_data *data, u32 addr, u32 *result); int (*write32)(struct tpm_tis_data *data, u32 addr, u32 src); + bool (*check_data)(struct tpm_tis_data *data, const u8 *buf, + size_t len); }; static inline int tpm_tis_read_bytes(struct tpm_tis_data *data, u32 addr, -- 2.7.4