soc: hisilicon: kunpeng_hccs: Support the platform with PCC type3 and interrupt ack
authorHuisong Li <lihuisong@huawei.com>
Fri, 1 Dec 2023 03:45:34 +0000 (11:45 +0800)
committerWei Xu <xuwei5@hisilicon.com>
Thu, 7 Dec 2023 06:16:35 +0000 (06:16 +0000)
Support the platform with PCC type3 and interrupt ack. And a version
specific structure is introduced to handle the difference between the
device in the code.

Signed-off-by: Huisong Li <lihuisong@huawei.com>
Reviewed-by: Jonathan Cameron <Jonathan.Cameron@huawei.com>
Signed-off-by: Wei Xu <xuwei5@hisilicon.com>
drivers/soc/hisilicon/kunpeng_hccs.c
drivers/soc/hisilicon/kunpeng_hccs.h

index 15125f1e0f3ea505ade3223eea461dfd42be2144..9ff70b38e5e99cab30efad847a6bb5ffe2374dd1 100644 (file)
@@ -110,6 +110,14 @@ static void hccs_chan_tx_done(struct mbox_client *cl, void *msg, int ret)
                         *(u8 *)msg, ret);
 }
 
+static void hccs_pcc_rx_callback(struct mbox_client *cl, void *mssg)
+{
+       struct hccs_mbox_client_info *cl_info =
+                       container_of(cl, struct hccs_mbox_client_info, client);
+
+       complete(&cl_info->done);
+}
+
 static void hccs_unregister_pcc_channel(struct hccs_dev *hdev)
 {
        struct hccs_mbox_client_info *cl_info = &hdev->cl_info;
@@ -131,6 +139,9 @@ static int hccs_register_pcc_channel(struct hccs_dev *hdev)
        cl->tx_block = false;
        cl->knows_txdone = true;
        cl->tx_done = hccs_chan_tx_done;
+       cl->rx_callback = hdev->verspec_data->rx_callback;
+       init_completion(&cl_info->done);
+
        pcc_chan = pcc_mbox_request_channel(cl, hdev->chan_id);
        if (IS_ERR(pcc_chan)) {
                dev_err(dev, "PPC channel request failed.\n");
@@ -147,10 +158,16 @@ static int hccs_register_pcc_channel(struct hccs_dev *hdev)
         */
        cl_info->deadline_us =
                        HCCS_PCC_CMD_WAIT_RETRIES_NUM * pcc_chan->latency;
-       if (cl_info->mbox_chan->mbox->txdone_irq) {
+       if (!hdev->verspec_data->has_txdone_irq &&
+           cl_info->mbox_chan->mbox->txdone_irq) {
                dev_err(dev, "PCC IRQ in PCCT is enabled.\n");
                rc = -EINVAL;
                goto err_mbx_channel_free;
+       } else if (hdev->verspec_data->has_txdone_irq &&
+                  !cl_info->mbox_chan->mbox->txdone_irq) {
+               dev_err(dev, "PCC IRQ in PCCT isn't supported.\n");
+               rc = -EINVAL;
+               goto err_mbx_channel_free;
        }
 
        if (pcc_chan->shmem_base_addr) {
@@ -172,7 +189,7 @@ out:
        return rc;
 }
 
-static int hccs_check_chan_cmd_complete(struct hccs_dev *hdev)
+static int hccs_wait_cmd_complete_by_poll(struct hccs_dev *hdev)
 {
        struct hccs_mbox_client_info *cl_info = &hdev->cl_info;
        struct acpi_pcct_shared_memory __iomem *comm_base =
@@ -194,30 +211,74 @@ static int hccs_check_chan_cmd_complete(struct hccs_dev *hdev)
        return ret;
 }
 
+static int hccs_wait_cmd_complete_by_irq(struct hccs_dev *hdev)
+{
+       struct hccs_mbox_client_info *cl_info = &hdev->cl_info;
+
+       if (!wait_for_completion_timeout(&cl_info->done,
+                       usecs_to_jiffies(cl_info->deadline_us))) {
+               dev_err(hdev->dev, "PCC command executed timeout!\n");
+               return -ETIMEDOUT;
+       }
+
+       return 0;
+}
+
+static inline void hccs_fill_pcc_shared_mem_region(struct hccs_dev *hdev,
+                                                  u8 cmd,
+                                                  struct hccs_desc *desc,
+                                                  void __iomem *comm_space,
+                                                  u16 space_size)
+{
+       struct acpi_pcct_shared_memory tmp = {
+               .signature = PCC_SIGNATURE | hdev->chan_id,
+               .command = cmd,
+               .status = 0,
+       };
+
+       memcpy_toio(hdev->cl_info.pcc_comm_addr, (void *)&tmp,
+                   sizeof(struct acpi_pcct_shared_memory));
+
+       /* Copy the message to the PCC comm space */
+       memcpy_toio(comm_space, (void *)desc, space_size);
+}
+
+static inline void hccs_fill_ext_pcc_shared_mem_region(struct hccs_dev *hdev,
+                                                      u8 cmd,
+                                                      struct hccs_desc *desc,
+                                                      void __iomem *comm_space,
+                                                      u16 space_size)
+{
+       struct acpi_pcct_ext_pcc_shared_memory tmp = {
+               .signature = PCC_SIGNATURE | hdev->chan_id,
+               .flags = PCC_CMD_COMPLETION_NOTIFY,
+               .length = HCCS_PCC_SHARE_MEM_BYTES,
+               .command = cmd,
+       };
+
+       memcpy_toio(hdev->cl_info.pcc_comm_addr, (void *)&tmp,
+                   sizeof(struct acpi_pcct_ext_pcc_shared_memory));
+
+       /* Copy the message to the PCC comm space */
+       memcpy_toio(comm_space, (void *)desc, space_size);
+}
+
 static int hccs_pcc_cmd_send(struct hccs_dev *hdev, u8 cmd,
                             struct hccs_desc *desc)
 {
+       const struct hccs_verspecific_data *verspec_data = hdev->verspec_data;
        struct hccs_mbox_client_info *cl_info = &hdev->cl_info;
-       void __iomem *comm_space = cl_info->pcc_comm_addr +
-                                       sizeof(struct acpi_pcct_shared_memory);
        struct hccs_fw_inner_head *fw_inner_head;
-       struct acpi_pcct_shared_memory tmp = {0};
-       u16 comm_space_size;
+       void __iomem *comm_space;
+       u16 space_size;
        int ret;
 
-       /* Write signature for this subspace */
-       tmp.signature = PCC_SIGNATURE | hdev->chan_id;
-       /* Write to the shared command region */
-       tmp.command = cmd;
-       /* Clear cmd complete bit */
-       tmp.status = 0;
-       memcpy_toio(cl_info->pcc_comm_addr, (void *)&tmp,
-                       sizeof(struct acpi_pcct_shared_memory));
-
-       /* Copy the message to the PCC comm space */
-       comm_space_size = HCCS_PCC_SHARE_MEM_BYTES -
-                               sizeof(struct acpi_pcct_shared_memory);
-       memcpy_toio(comm_space, (void *)desc, comm_space_size);
+       comm_space = cl_info->pcc_comm_addr + verspec_data->shared_mem_size;
+       space_size = HCCS_PCC_SHARE_MEM_BYTES - verspec_data->shared_mem_size;
+       verspec_data->fill_pcc_shared_mem(hdev, cmd, desc,
+                                         comm_space, space_size);
+       if (verspec_data->has_txdone_irq)
+               reinit_completion(&cl_info->done);
 
        /* Ring doorbell */
        ret = mbox_send_message(cl_info->mbox_chan, &cmd);
@@ -227,13 +288,12 @@ static int hccs_pcc_cmd_send(struct hccs_dev *hdev, u8 cmd,
                goto end;
        }
 
-       /* Wait for completion */
-       ret = hccs_check_chan_cmd_complete(hdev);
+       ret = verspec_data->wait_cmd_complete(hdev);
        if (ret)
                goto end;
 
        /* Copy response data */
-       memcpy_fromio((void *)desc, comm_space, comm_space_size);
+       memcpy_fromio((void *)desc, comm_space, space_size);
        fw_inner_head = &desc->rsp.fw_inner_head;
        if (fw_inner_head->retStatus) {
                dev_err(hdev->dev, "Execute PCC command failed, error code = %u.\n",
@@ -242,7 +302,10 @@ static int hccs_pcc_cmd_send(struct hccs_dev *hdev, u8 cmd,
        }
 
 end:
-       mbox_client_txdone(cl_info->mbox_chan, ret);
+       if (verspec_data->has_txdone_irq)
+               mbox_chan_txdone(cl_info->mbox_chan, ret);
+       else
+               mbox_client_txdone(cl_info->mbox_chan, ret);
        return ret;
 }
 
@@ -1213,6 +1276,11 @@ static int hccs_probe(struct platform_device *pdev)
        hdev->dev = &pdev->dev;
        platform_set_drvdata(pdev, hdev);
 
+       /*
+        * Here would never be failure as the driver and device has been matched.
+        */
+       hdev->verspec_data = acpi_device_get_match_data(hdev->dev);
+
        mutex_init(&hdev->lock);
        rc = hccs_get_pcc_chan_id(hdev);
        if (rc)
@@ -1249,9 +1317,26 @@ static void hccs_remove(struct platform_device *pdev)
        hccs_unregister_pcc_channel(hdev);
 }
 
+static const struct hccs_verspecific_data hisi04b1_verspec_data = {
+       .rx_callback = NULL,
+       .wait_cmd_complete = hccs_wait_cmd_complete_by_poll,
+       .fill_pcc_shared_mem = hccs_fill_pcc_shared_mem_region,
+       .shared_mem_size = sizeof(struct acpi_pcct_shared_memory),
+       .has_txdone_irq = false,
+};
+
+static const struct hccs_verspecific_data hisi04b2_verspec_data = {
+       .rx_callback = hccs_pcc_rx_callback,
+       .wait_cmd_complete = hccs_wait_cmd_complete_by_irq,
+       .fill_pcc_shared_mem = hccs_fill_ext_pcc_shared_mem_region,
+       .shared_mem_size = sizeof(struct acpi_pcct_ext_pcc_shared_memory),
+       .has_txdone_irq = true,
+};
+
 static const struct acpi_device_id hccs_acpi_match[] = {
-       { "HISI04B1"},
-       { ""},
+       { "HISI04B1", (unsigned long)&hisi04b1_verspec_data},
+       { "HISI04B2", (unsigned long)&hisi04b2_verspec_data},
+       { }
 };
 MODULE_DEVICE_TABLE(acpi, hccs_acpi_match);
 
index 6012d27760281669030cb85739db2d0e5f9a79f8..c3adbc01b4719b7cb3ec866698c2ba1f2bbf079a 100644 (file)
@@ -51,11 +51,26 @@ struct hccs_mbox_client_info {
        struct pcc_mbox_chan *pcc_chan;
        u64 deadline_us;
        void __iomem *pcc_comm_addr;
+       struct completion done;
+};
+
+struct hccs_desc;
+
+struct hccs_verspecific_data {
+       void (*rx_callback)(struct mbox_client *cl, void *mssg);
+       int (*wait_cmd_complete)(struct hccs_dev *hdev);
+       void (*fill_pcc_shared_mem)(struct hccs_dev *hdev,
+                                   u8 cmd, struct hccs_desc *desc,
+                                   void __iomem *comm_space,
+                                   u16 space_size);
+       u16 shared_mem_size;
+       bool has_txdone_irq;
 };
 
 struct hccs_dev {
        struct device *dev;
        struct acpi_device *acpi_dev;
+       const struct hccs_verspecific_data *verspec_data;
        u64 caps;
        u8 chip_num;
        struct hccs_chip_info *chips;