linux-kernel.vger.kernel.org archive mirror
 help / color / mirror / Atom feed
From: Ezequiel Garcia <ezequiel@collabora.com>
To: Jarkko Sakkinen <jarkko@kernel.org>
Cc: Adrian Ratiu <adrian.ratiu@collabora.com>,
	linux-integrity@vger.kernel.org, Peter Huewe <peterhuewe@gmx.de>,
	Jason Gunthorpe <jgg@ziepe.ca>,
	Helen Koike <helen.koike@collabora.com>,
	Duncan Laurie <dlaurie@chromium.org>,
	Stephen Boyd <swboyd@chromium.org>,
	kernel@collabora.com, linux-kernel@vger.kernel.org
Subject: Re: [PATCH v2] char: tpm: add i2c driver for cr50
Date: Sun, 29 Nov 2020 11:51:07 -0300	[thread overview]
Message-ID: <cbbb87015574eb06c7a9d99bd5757b257e0aa36b.camel@collabora.com> (raw)
In-Reply-To: <20201129032933.GE39488@kernel.org>

On Sun, 2020-11-29 at 05:29 +0200, Jarkko Sakkinen wrote:
> On Thu, Nov 26, 2020 at 03:19:24AM -0300, Ezequiel Garcia wrote:
> > On Thu, 2020-11-26 at 05:30 +0200, Jarkko Sakkinen wrote:
> > > On Tue, 2020-11-24 at 10:14 -0300, Ezequiel Garcia wrote:
> > > > Hi Jarkko,
> > > > 
> > > > Thanks for your review.
> > > > 
> > > > On Tue, 2020-11-24 at 00:06 +0200, Jarkko Sakkinen wrote:
> > > > > On Fri, Nov 20, 2020 at 07:23:45PM +0200, Adrian Ratiu wrote:
> > > > > > From: "dlaurie@chromium.org" <dlaurie@chromium.org>
> > > > > > 
> > > > > > Add TPM 2.0 compatible I2C interface for chips with cr50
> > > > > > firmware.
> > > > > > 
> > > > > > The firmware running on the currently supported H1 MCU requires a
> > > > > > special driver to handle its specific protocol, and this makes it
> > > > > > unsuitable to use tpm_tis_core_* and instead it must implement
> > > > > > the
> > > > > > underlying TPM protocol similar to the other I2C TPM drivers.
> > > > > > 
> > > > > > - All 4 byes of status register must be read/written at once.
> > > > > > - FIFO and burst count is limited to 63 and must be drained by
> > > > > > AP.
> > > > > > - Provides an interrupt to indicate when read response data is
> > > > > > ready
> > > > > > and when the TPM is finished processing write data.
> > > > > > 
> > > > > > This driver is based on the existing infineon I2C TPM driver,
> > > > > > which
> > > > > > most closely matches the cr50 i2c protocol behavior.
> > > > > > 
> > > > > > Cc: Helen Koike <helen.koike@collabora.com>
> > > > > > Signed-off-by: Duncan Laurie <dlaurie@chromium.org>
> > > > > > [swboyd@chromium.org: Depend on i2c even if it's a module,
> > > > > > replace
> > > > > > boilier plate with SPDX tag, drop asm/byteorder.h include,
> > > > > > simplify
> > > > > > return from probe]
> > > > > > Signed-off-by: Stephen Boyd <swboyd@chromium.org>
> > > > > > Signed-off-by: Fabien Lahoudere <fabien.lahoudere@collabora.com>
> > > > > > Signed-off-by: Adrian Ratiu <adrian.ratiu@collabora.com>
> > > > > > ---
> > > > > > Changes in v2:
> > > > > >   - Various small fixes all over (reorder includes, MAX_BUFSIZE,
> > > > > > comments, etc)
> > > > > >   - Reworked return values of i2c_wait_tpm_ready() to fix timeout
> > > > > > mis-handling
> > > > > > so ret == 0 now means success, the wait period jiffies is ignored
> > > > > > because that
> > > > > > number is meaningless and return a proper timeout error in case
> > > > > > jiffies == 0.
> > > > > >   - Make i2c default to 1 message per transfer (requested by
> > > > > > Helen)
> > > > > >   - Move -EIO error reporting to transfer function to cleanup
> > > > > > transfer() itself
> > > > > > and its R/W callers
> > > > > >   - Remove magic value hardcodings and introduce enum
> > > > > > force_release.
> > > > > > 
> > > > > > v1 posted at https://lkml.org/lkml/2020/2/25/349
> > > > > > 
> > > > > > Applies on next-20201120, tested on Chromebook EVE.
> > > > > > ---
> > > > > >  drivers/char/tpm/Kconfig            |  10 +
> > > > > >  drivers/char/tpm/Makefile           |   2 +
> > > > > >  drivers/char/tpm/tpm_tis_i2c_cr50.c | 768
> > > > > > ++++++++++++++++++++++++++++
> > > > > >  3 files changed, 780 insertions(+)
> > > > > >  create mode 100644 drivers/char/tpm/tpm_tis_i2c_cr50.c
> > > > > > 
> > > > > > diff --git a/drivers/char/tpm/Kconfig b/drivers/char/tpm/Kconfig
> > > > > > index a18c314da211..4308f9ca7a43 100644
> > > > > > --- a/drivers/char/tpm/Kconfig
> > > > > > +++ b/drivers/char/tpm/Kconfig
> > > > > > @@ -86,6 +86,16 @@ config TCG_TIS_SYNQUACER
> > > > > >           To compile this driver as a module, choose  M here;
> > > > > >           the module will be called tpm_tis_synquacer.
> > > > > >  
> > > > > > +config TCG_TIS_I2C_CR50
> > > > > > +       tristate "TPM Interface Specification 2.0 Interface (I2C
> > > > > > - CR50)"
> > > > > > +       depends on I2C
> > > > > > +       select TCG_CR50
> > > > > > +       help
> > > > > > +         This is a driver for the Google cr50 I2C TPM interface
> > > > > > which is a
> > > > > > +         custom microcontroller and requires a custom i2c
> > > > > > protocol interface
> > > > > > +         to handle the limitations of the hardware.  To compile
> > > > > > this driver
> > > > > > +         as a module, choose M here; the module will be called
> > > > > > tcg_tis_i2c_cr50.
> > > > > > +
> > > > > >  config TCG_TIS_I2C_ATMEL
> > > > > >         tristate "TPM Interface Specification 1.2 Interface (I2C
> > > > > > - Atmel)"
> > > > > >         depends on I2C
> > > > > > diff --git a/drivers/char/tpm/Makefile
> > > > > > b/drivers/char/tpm/Makefile
> > > > > > index 84db4fb3a9c9..66d39ea6bd10 100644
> > > > > > --- a/drivers/char/tpm/Makefile
> > > > > > +++ b/drivers/char/tpm/Makefile
> > > > > > @@ -27,6 +27,8 @@ obj-$(CONFIG_TCG_TIS_SPI) += tpm_tis_spi.o
> > > > > >  tpm_tis_spi-y := tpm_tis_spi_main.o
> > > > > >  tpm_tis_spi-$(CONFIG_TCG_TIS_SPI_CR50) += tpm_tis_spi_cr50.o
> > > > > >  
> > > > > > +obj-$(CONFIG_TCG_TIS_I2C_CR50) += tpm_tis_i2c_cr50.o
> > > > > > +
> > > > > >  obj-$(CONFIG_TCG_TIS_I2C_ATMEL) += tpm_i2c_atmel.o
> > > > > >  obj-$(CONFIG_TCG_TIS_I2C_INFINEON) += tpm_i2c_infineon.o
> > > > > >  obj-$(CONFIG_TCG_TIS_I2C_NUVOTON) += tpm_i2c_nuvoton.o
> > > > > > diff --git a/drivers/char/tpm/tpm_tis_i2c_cr50.c
> > > > > > b/drivers/char/tpm/tpm_tis_i2c_cr50.c
> > > > > > new file mode 100644
> > > > > > index 000000000000..37555dafdca0
> > > > > > --- /dev/null
> > > > > > +++ b/drivers/char/tpm/tpm_tis_i2c_cr50.c
> > > > > > @@ -0,0 +1,768 @@
> > > > > > +// SPDX-License-Identifier: GPL-2.0
> > > > > > +/*
> > > > > > + * Copyright 2016 Google Inc.
> > > > > > + *
> > > > > > + * Based on Linux Kernel TPM driver by
> > > > > > + * Peter Huewe <peter.huewe@infineon.com>
> > > > > > + * Copyright (C) 2011 Infineon Technologies
> > > > > > + */
> > > > > > +
> > > > > > +/*
> > > > > > + * cr50 is a firmware for H1 secure modules that requires
> > > > > > special
> > > > > > + * handling for the I2C interface.
> > > > > > + *
> > > > > > + * - Use an interrupt for transaction status instead of
> > > > > > hardcoded delays
> > > > > > + * - Must use write+wait+read read protocol
> > > > > > + * - All 4 bytes of status register must be read/written at once
> > > > > > + * - Burst count max is 63 bytes, and burst count behaves
> > > > > > + *   slightly differently than other I2C TPMs
> > > > > > + * - When reading from FIFO the full burstcnt must be read
> > > > > > + *   instead of just reading header and determining the
> > > > > > remainder
> > > > > > + */
> > > > > > +
> > > > > > +#include <linux/acpi.h>
> > > > > > +#include <linux/completion.h>
> > > > > > +#include <linux/i2c.h>
> > > > > > +#include <linux/interrupt.h>
> > > > > > +#include <linux/module.h>
> > > > > > +#include <linux/pm.h>
> > > > > > +#include <linux/slab.h>
> > > > > > +#include <linux/wait.h>
> > > > > > +
> > > > > > +#include "tpm_tis_core.h"
> > > > > > +
> > > > > > +#define CR50_MAX_BUFSIZE       64
> > > > > > +#define CR50_TIMEOUT_SHORT_MS  2       /* Short timeout during
> > > > > > transactions */
> > > > > > +#define CR50_TIMEOUT_NOIRQ_MS  20      /* Timeout for TPM ready
> > > > > > without IRQ */
> > > > > > +#define CR50_I2C_DID_VID       0x00281ae0L
> > > > > > +#define CR50_I2C_MAX_RETRIES   3       /* Max retries due to I2C
> > > > > > errors */
> > > > > > +#define CR50_I2C_RETRY_DELAY_LO        55      /* Min usecs
> > > > > > between retries on I2C */
> > > > > > +#define CR50_I2C_RETRY_DELAY_HI        65      /* Max usecs
> > > > > > between retries on I2C */
> > > > > 
> > > > > CR50_ -> TPM_CR50_
> > > > > 
> > > > > > +
> > > > > > +#define TPM_I2C_ACCESS(l)      (0x0000 | ((l) << 4))
> > > > > > +#define TPM_I2C_STS(l)         (0x0001 | ((l) << 4))
> > > > > > +#define TPM_I2C_DATA_FIFO(l)   (0x0005 | ((l) << 4))
> > > > > > +#define TPM_I2C_DID_VID(l)     (0x0006 | ((l) << 4))
> > > > > > +
> > > > > > +struct priv_data {
> > > > > > +       int irq;
> > > > > > +       int locality;
> > > > > > +       struct completion tpm_ready;
> > > > > > +       u8 buf[CR50_MAX_BUFSIZE];
> > > > > > +};
> > > > > 
> > > > > tpm_i2c_cr50_priv_data
> > > > > 
> > > > > > +
> > > > > > +enum force_release {
> > > > > > +       CR50_NO_FORCE = 0x0,
> > > > > > +       CR50_FORCE = 0x1,
> > > > > > +};
> > > > > 
> > > > > I'd just 
> > > > > 
> > > > > #define TPM_I2C_CR50_NO_FORCE   0
> > > > > #define TPM_I2C_CR50_FORCE      1
> > > > > 
> > > > 
> > > > A proper enumerated type has advantages over a preprocessor macro:
> > > > even if the compiler won't warn you, static analyzers can warn
> > > > about a misuse.
> > > 
> > > Why don't you just use "bool", "true" and "false"? I ignored that
> > > this has nothing to do with the hardware last time.
> > > 
> > 
> > Well, boolean parameters are a known anti-pattern [1].
> > 
> > [1] https://people.mpi-inf.mpg.de/~jblanche/api-design.pdf
> 
> It's an internal function to begin with.
> 

Too bad you see it that way. Anti-patterns spread like the flu,
and then it takes a lot of energy to take care of them.

Cheers,
Ezequiel


      reply	other threads:[~2020-11-29 14:52 UTC|newest]

Thread overview: 11+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
2020-11-20 17:23 [PATCH v2] char: tpm: add i2c driver for cr50 Adrian Ratiu
2020-11-20 19:11 ` Helen Koike
2020-11-23 17:41   ` Adrian Ratiu
2020-11-23 22:06 ` Jarkko Sakkinen
2020-11-24 12:24   ` Adrian Ratiu
2020-11-24 13:14   ` Ezequiel Garcia
2020-11-26  3:30     ` Jarkko Sakkinen
2020-11-26  6:19       ` Ezequiel Garcia
2020-11-26 10:30         ` Adrian Ratiu
2020-11-29  3:29         ` Jarkko Sakkinen
2020-11-29 14:51           ` Ezequiel Garcia [this message]

Reply instructions:

You may reply publicly to this message via plain-text email
using any one of the following methods:

* Save the following mbox file, import it into your mail client,
  and reply-to-all from there: mbox

  Avoid top-posting and favor interleaved quoting:
  https://en.wikipedia.org/wiki/Posting_style#Interleaved_style

* Reply using the --to, --cc, and --in-reply-to
  switches of git-send-email(1):

  git send-email \
    --in-reply-to=cbbb87015574eb06c7a9d99bd5757b257e0aa36b.camel@collabora.com \
    --to=ezequiel@collabora.com \
    --cc=adrian.ratiu@collabora.com \
    --cc=dlaurie@chromium.org \
    --cc=helen.koike@collabora.com \
    --cc=jarkko@kernel.org \
    --cc=jgg@ziepe.ca \
    --cc=kernel@collabora.com \
    --cc=linux-integrity@vger.kernel.org \
    --cc=linux-kernel@vger.kernel.org \
    --cc=peterhuewe@gmx.de \
    --cc=swboyd@chromium.org \
    /path/to/YOUR_REPLY

  https://kernel.org/pub/software/scm/git/docs/git-send-email.html

* If your mail client supports setting the In-Reply-To header
  via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).