mirror of https://gitee.com/openkylin/linux.git
tpm_crb: map locality registers
In order to provide access to locality registers, this commits adds mapping of the head of the CRB registers, which are located right before the control area. Signed-off-by: Jarkko Sakkinen <jarkko.sakkinen@linux.intel.com> Reviewed-by: Jerry Snitselaar <jsnitsel@redhat.com> Tested-by: Gang Wei <gang.wei@intel.com>
This commit is contained in:
parent
5cc0101d1f
commit
13b1f4a571
|
@ -52,18 +52,28 @@ enum crb_cancel {
|
||||||
CRB_CANCEL_INVOKE = BIT(0),
|
CRB_CANCEL_INVOKE = BIT(0),
|
||||||
};
|
};
|
||||||
|
|
||||||
struct crb_control_area {
|
struct crb_regs_head {
|
||||||
u32 req;
|
u32 loc_state;
|
||||||
u32 sts;
|
u32 reserved1;
|
||||||
u32 cancel;
|
u32 loc_ctrl;
|
||||||
u32 start;
|
u32 loc_sts;
|
||||||
u32 int_enable;
|
u8 reserved2[32];
|
||||||
u32 int_sts;
|
u64 intf_id;
|
||||||
u32 cmd_size;
|
u64 ctrl_ext;
|
||||||
u32 cmd_pa_low;
|
} __packed;
|
||||||
u32 cmd_pa_high;
|
|
||||||
u32 rsp_size;
|
struct crb_regs_tail {
|
||||||
u64 rsp_pa;
|
u32 ctrl_req;
|
||||||
|
u32 ctrl_sts;
|
||||||
|
u32 ctrl_cancel;
|
||||||
|
u32 ctrl_start;
|
||||||
|
u32 ctrl_int_enable;
|
||||||
|
u32 ctrl_int_sts;
|
||||||
|
u32 ctrl_cmd_size;
|
||||||
|
u32 ctrl_cmd_pa_low;
|
||||||
|
u32 ctrl_cmd_pa_high;
|
||||||
|
u32 ctrl_rsp_size;
|
||||||
|
u64 ctrl_rsp_pa;
|
||||||
} __packed;
|
} __packed;
|
||||||
|
|
||||||
enum crb_status {
|
enum crb_status {
|
||||||
|
@ -78,7 +88,8 @@ enum crb_flags {
|
||||||
struct crb_priv {
|
struct crb_priv {
|
||||||
unsigned int flags;
|
unsigned int flags;
|
||||||
void __iomem *iobase;
|
void __iomem *iobase;
|
||||||
struct crb_control_area __iomem *cca;
|
struct crb_regs_head __iomem *regs_h;
|
||||||
|
struct crb_regs_tail __iomem *regs_t;
|
||||||
u8 __iomem *cmd;
|
u8 __iomem *cmd;
|
||||||
u8 __iomem *rsp;
|
u8 __iomem *rsp;
|
||||||
u32 cmd_size;
|
u32 cmd_size;
|
||||||
|
@ -104,7 +115,7 @@ static int __maybe_unused crb_go_idle(struct device *dev, struct crb_priv *priv)
|
||||||
if (priv->flags & CRB_FL_ACPI_START)
|
if (priv->flags & CRB_FL_ACPI_START)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
iowrite32(CRB_CTRL_REQ_GO_IDLE, &priv->cca->req);
|
iowrite32(CRB_CTRL_REQ_GO_IDLE, &priv->regs_t->ctrl_req);
|
||||||
/* we don't really care when this settles */
|
/* we don't really care when this settles */
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -128,21 +139,23 @@ static int __maybe_unused crb_cmd_ready(struct device *dev,
|
||||||
struct crb_priv *priv)
|
struct crb_priv *priv)
|
||||||
{
|
{
|
||||||
ktime_t stop, start;
|
ktime_t stop, start;
|
||||||
|
u32 req;
|
||||||
|
|
||||||
if (priv->flags & CRB_FL_ACPI_START)
|
if (priv->flags & CRB_FL_ACPI_START)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
iowrite32(CRB_CTRL_REQ_CMD_READY, &priv->cca->req);
|
iowrite32(CRB_CTRL_REQ_CMD_READY, &priv->regs_t->ctrl_req);
|
||||||
|
|
||||||
start = ktime_get();
|
start = ktime_get();
|
||||||
stop = ktime_add(start, ms_to_ktime(TPM2_TIMEOUT_C));
|
stop = ktime_add(start, ms_to_ktime(TPM2_TIMEOUT_C));
|
||||||
do {
|
do {
|
||||||
if (!(ioread32(&priv->cca->req) & CRB_CTRL_REQ_CMD_READY))
|
req = ioread32(&priv->regs_t->ctrl_req);
|
||||||
|
if (!(req & CRB_CTRL_REQ_CMD_READY))
|
||||||
return 0;
|
return 0;
|
||||||
usleep_range(50, 100);
|
usleep_range(50, 100);
|
||||||
} while (ktime_before(ktime_get(), stop));
|
} while (ktime_before(ktime_get(), stop));
|
||||||
|
|
||||||
if (ioread32(&priv->cca->req) & CRB_CTRL_REQ_CMD_READY) {
|
if (ioread32(&priv->regs_t->ctrl_req) & CRB_CTRL_REQ_CMD_READY) {
|
||||||
dev_warn(dev, "cmdReady timed out\n");
|
dev_warn(dev, "cmdReady timed out\n");
|
||||||
return -ETIME;
|
return -ETIME;
|
||||||
}
|
}
|
||||||
|
@ -155,7 +168,7 @@ static u8 crb_status(struct tpm_chip *chip)
|
||||||
struct crb_priv *priv = dev_get_drvdata(&chip->dev);
|
struct crb_priv *priv = dev_get_drvdata(&chip->dev);
|
||||||
u8 sts = 0;
|
u8 sts = 0;
|
||||||
|
|
||||||
if ((ioread32(&priv->cca->start) & CRB_START_INVOKE) !=
|
if ((ioread32(&priv->regs_t->ctrl_start) & CRB_START_INVOKE) !=
|
||||||
CRB_START_INVOKE)
|
CRB_START_INVOKE)
|
||||||
sts |= CRB_DRV_STS_COMPLETE;
|
sts |= CRB_DRV_STS_COMPLETE;
|
||||||
|
|
||||||
|
@ -171,7 +184,7 @@ static int crb_recv(struct tpm_chip *chip, u8 *buf, size_t count)
|
||||||
if (count < 6)
|
if (count < 6)
|
||||||
return -EIO;
|
return -EIO;
|
||||||
|
|
||||||
if (ioread32(&priv->cca->sts) & CRB_CTRL_STS_ERROR)
|
if (ioread32(&priv->regs_t->ctrl_sts) & CRB_CTRL_STS_ERROR)
|
||||||
return -EIO;
|
return -EIO;
|
||||||
|
|
||||||
memcpy_fromio(buf, priv->rsp, 6);
|
memcpy_fromio(buf, priv->rsp, 6);
|
||||||
|
@ -210,7 +223,7 @@ static int crb_send(struct tpm_chip *chip, u8 *buf, size_t len)
|
||||||
/* Zero the cancel register so that the next command will not get
|
/* Zero the cancel register so that the next command will not get
|
||||||
* canceled.
|
* canceled.
|
||||||
*/
|
*/
|
||||||
iowrite32(0, &priv->cca->cancel);
|
iowrite32(0, &priv->regs_t->ctrl_cancel);
|
||||||
|
|
||||||
if (len > priv->cmd_size) {
|
if (len > priv->cmd_size) {
|
||||||
dev_err(&chip->dev, "invalid command count value %zd %d\n",
|
dev_err(&chip->dev, "invalid command count value %zd %d\n",
|
||||||
|
@ -224,7 +237,7 @@ static int crb_send(struct tpm_chip *chip, u8 *buf, size_t len)
|
||||||
wmb();
|
wmb();
|
||||||
|
|
||||||
if (priv->flags & CRB_FL_CRB_START)
|
if (priv->flags & CRB_FL_CRB_START)
|
||||||
iowrite32(CRB_START_INVOKE, &priv->cca->start);
|
iowrite32(CRB_START_INVOKE, &priv->regs_t->ctrl_start);
|
||||||
|
|
||||||
if (priv->flags & CRB_FL_ACPI_START)
|
if (priv->flags & CRB_FL_ACPI_START)
|
||||||
rc = crb_do_acpi_start(chip);
|
rc = crb_do_acpi_start(chip);
|
||||||
|
@ -236,7 +249,7 @@ static void crb_cancel(struct tpm_chip *chip)
|
||||||
{
|
{
|
||||||
struct crb_priv *priv = dev_get_drvdata(&chip->dev);
|
struct crb_priv *priv = dev_get_drvdata(&chip->dev);
|
||||||
|
|
||||||
iowrite32(CRB_CANCEL_INVOKE, &priv->cca->cancel);
|
iowrite32(CRB_CANCEL_INVOKE, &priv->regs_t->ctrl_cancel);
|
||||||
|
|
||||||
if ((priv->flags & CRB_FL_ACPI_START) && crb_do_acpi_start(chip))
|
if ((priv->flags & CRB_FL_ACPI_START) && crb_do_acpi_start(chip))
|
||||||
dev_err(&chip->dev, "ACPI Start failed\n");
|
dev_err(&chip->dev, "ACPI Start failed\n");
|
||||||
|
@ -245,7 +258,7 @@ static void crb_cancel(struct tpm_chip *chip)
|
||||||
static bool crb_req_canceled(struct tpm_chip *chip, u8 status)
|
static bool crb_req_canceled(struct tpm_chip *chip, u8 status)
|
||||||
{
|
{
|
||||||
struct crb_priv *priv = dev_get_drvdata(&chip->dev);
|
struct crb_priv *priv = dev_get_drvdata(&chip->dev);
|
||||||
u32 cancel = ioread32(&priv->cca->cancel);
|
u32 cancel = ioread32(&priv->regs_t->ctrl_cancel);
|
||||||
|
|
||||||
return (cancel & CRB_CANCEL_INVOKE) == CRB_CANCEL_INVOKE;
|
return (cancel & CRB_CANCEL_INVOKE) == CRB_CANCEL_INVOKE;
|
||||||
}
|
}
|
||||||
|
@ -345,10 +358,22 @@ static int crb_map_io(struct acpi_device *device, struct crb_priv *priv,
|
||||||
if (IS_ERR(priv->iobase))
|
if (IS_ERR(priv->iobase))
|
||||||
return PTR_ERR(priv->iobase);
|
return PTR_ERR(priv->iobase);
|
||||||
|
|
||||||
priv->cca = crb_map_res(dev, priv, &io_res, buf->control_address,
|
/* The ACPI IO region starts at the head area and continues to include
|
||||||
sizeof(struct crb_control_area));
|
* the control area, as one nice sane region except for some older
|
||||||
if (IS_ERR(priv->cca))
|
* stuff that puts the control area outside the ACPI IO region.
|
||||||
return PTR_ERR(priv->cca);
|
*/
|
||||||
|
if (!(priv->flags & CRB_FL_ACPI_START)) {
|
||||||
|
if (buf->control_address == io_res.start +
|
||||||
|
sizeof(*priv->regs_h))
|
||||||
|
priv->regs_h = priv->iobase;
|
||||||
|
else
|
||||||
|
dev_warn(dev, FW_BUG "Bad ACPI memory layout");
|
||||||
|
}
|
||||||
|
|
||||||
|
priv->regs_t = crb_map_res(dev, priv, &io_res, buf->control_address,
|
||||||
|
sizeof(struct crb_regs_tail));
|
||||||
|
if (IS_ERR(priv->regs_t))
|
||||||
|
return PTR_ERR(priv->regs_t);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* PTT HW bug w/a: wake up the device to access
|
* PTT HW bug w/a: wake up the device to access
|
||||||
|
@ -358,11 +383,11 @@ static int crb_map_io(struct acpi_device *device, struct crb_priv *priv,
|
||||||
if (ret)
|
if (ret)
|
||||||
return ret;
|
return ret;
|
||||||
|
|
||||||
pa_high = ioread32(&priv->cca->cmd_pa_high);
|
pa_high = ioread32(&priv->regs_t->ctrl_cmd_pa_high);
|
||||||
pa_low = ioread32(&priv->cca->cmd_pa_low);
|
pa_low = ioread32(&priv->regs_t->ctrl_cmd_pa_low);
|
||||||
cmd_pa = ((u64)pa_high << 32) | pa_low;
|
cmd_pa = ((u64)pa_high << 32) | pa_low;
|
||||||
cmd_size = crb_fixup_cmd_size(dev, &io_res, cmd_pa,
|
cmd_size = crb_fixup_cmd_size(dev, &io_res, cmd_pa,
|
||||||
ioread32(&priv->cca->cmd_size));
|
ioread32(&priv->regs_t->ctrl_cmd_size));
|
||||||
|
|
||||||
dev_dbg(dev, "cmd_hi = %X cmd_low = %X cmd_size %X\n",
|
dev_dbg(dev, "cmd_hi = %X cmd_low = %X cmd_size %X\n",
|
||||||
pa_high, pa_low, cmd_size);
|
pa_high, pa_low, cmd_size);
|
||||||
|
@ -373,10 +398,10 @@ static int crb_map_io(struct acpi_device *device, struct crb_priv *priv,
|
||||||
goto out;
|
goto out;
|
||||||
}
|
}
|
||||||
|
|
||||||
memcpy_fromio(&rsp_pa, &priv->cca->rsp_pa, 8);
|
memcpy_fromio(&rsp_pa, &priv->regs_t->ctrl_rsp_pa, 8);
|
||||||
rsp_pa = le64_to_cpu(rsp_pa);
|
rsp_pa = le64_to_cpu(rsp_pa);
|
||||||
rsp_size = crb_fixup_cmd_size(dev, &io_res, rsp_pa,
|
rsp_size = crb_fixup_cmd_size(dev, &io_res, rsp_pa,
|
||||||
ioread32(&priv->cca->rsp_size));
|
ioread32(&priv->regs_t->ctrl_rsp_size));
|
||||||
|
|
||||||
if (cmd_pa != rsp_pa) {
|
if (cmd_pa != rsp_pa) {
|
||||||
priv->rsp = crb_map_res(dev, priv, &io_res, rsp_pa, rsp_size);
|
priv->rsp = crb_map_res(dev, priv, &io_res, rsp_pa, rsp_size);
|
||||||
|
|
Loading…
Reference in New Issue