]> git.ipfire.org Git - thirdparty/kernel/linux.git/commitdiff
iommu/arm-smmu-v3: Use the HW arm_smmu_cmd in cmdq selection functions
authorJason Gunthorpe <jgg@nvidia.com>
Wed, 13 May 2026 23:57:41 +0000 (20:57 -0300)
committerWill Deacon <will@kernel.org>
Tue, 19 May 2026 14:06:12 +0000 (15:06 +0100)
Start removing the use of struct arm_smmu_cmdq_ent, starting at the
lower levels of the call chain. Change the functions that determine what
cmdq to issue the batch to into using struct arm_smmu_cmd directly.

Reviewed-by: Pranjal Shrivastava <praan@google.com>
Reviewed-by: Mostafa Saleh <smostafa@google.com>
Tested-by: Pranjal Shrivastava <praan@google.com>
Tested-by: Mostafa Saleh <smostafa@google.com>
Signed-off-by: Jason Gunthorpe <jgg@nvidia.com>
Reviewed-by: Nicolin Chen <nicolinc@nvidia.com>
Tested-by: Samiullah Khawaja <skhawaja@google.com>
Tested-by: Nicolin Chen <nicolinc@nvidia.com>
Signed-off-by: Will Deacon <will@kernel.org>
drivers/iommu/arm/arm-smmu-v3/arm-smmu-v3.c
drivers/iommu/arm/arm-smmu-v3/arm-smmu-v3.h
drivers/iommu/arm/arm-smmu-v3/tegra241-cmdqv.c

index 4a0b15b1d4d244e26c9c874f99cda372b137baf4..866471c7684f26bde14cb11f8a98bd0e2f24e1da 100644 (file)
@@ -371,12 +371,12 @@ static int arm_smmu_cmdq_build_cmd(u64 *cmd, struct arm_smmu_cmdq_ent *ent)
 }
 
 static struct arm_smmu_cmdq *arm_smmu_get_cmdq(struct arm_smmu_device *smmu,
-                                              struct arm_smmu_cmdq_ent *ent)
+                                              struct arm_smmu_cmd *cmd)
 {
        struct arm_smmu_cmdq *cmdq = NULL;
 
        if (smmu->impl_ops && smmu->impl_ops->get_secondary_cmdq)
-               cmdq = smmu->impl_ops->get_secondary_cmdq(smmu, ent);
+               cmdq = smmu->impl_ops->get_secondary_cmdq(smmu, cmd);
 
        return cmdq ?: &smmu->cmdq;
 }
@@ -924,16 +924,16 @@ static int __arm_smmu_cmdq_issue_cmd(struct arm_smmu_device *smmu,
                                     struct arm_smmu_cmdq_ent *ent,
                                     bool sync)
 {
-       u64 cmd[CMDQ_ENT_DWORDS];
+       struct arm_smmu_cmd cmd;
 
-       if (unlikely(arm_smmu_cmdq_build_cmd(cmd, ent))) {
+       if (unlikely(arm_smmu_cmdq_build_cmd(cmd.data, ent))) {
                dev_warn(smmu->dev, "ignoring unknown CMDQ opcode 0x%x\n",
                         ent->opcode);
                return -EINVAL;
        }
 
        return arm_smmu_cmdq_issue_cmdlist(
-               smmu, arm_smmu_get_cmdq(smmu, ent), cmd, 1, sync);
+               smmu, arm_smmu_get_cmdq(smmu, &cmd), cmd.data, 1, sync);
 }
 
 static int arm_smmu_cmdq_issue_cmd(struct arm_smmu_device *smmu,
@@ -948,42 +948,55 @@ static int arm_smmu_cmdq_issue_cmd_with_sync(struct arm_smmu_device *smmu,
        return __arm_smmu_cmdq_issue_cmd(smmu, ent, true);
 }
 
+static void arm_smmu_cmdq_batch_init_cmd(struct arm_smmu_device *smmu,
+                                        struct arm_smmu_cmdq_batch *cmds,
+                                        struct arm_smmu_cmd *cmd)
+{
+       cmds->num = 0;
+       cmds->cmdq = arm_smmu_get_cmdq(smmu, cmd);
+}
+
 static void arm_smmu_cmdq_batch_init(struct arm_smmu_device *smmu,
                                     struct arm_smmu_cmdq_batch *cmds,
                                     struct arm_smmu_cmdq_ent *ent)
 {
-       cmds->num = 0;
-       cmds->cmdq = arm_smmu_get_cmdq(smmu, ent);
+       struct arm_smmu_cmd cmd;
+
+       arm_smmu_cmdq_build_cmd(cmd.data, ent);
+       arm_smmu_cmdq_batch_init_cmd(smmu, cmds, &cmd);
 }
 
 static void arm_smmu_cmdq_batch_add(struct arm_smmu_device *smmu,
                                    struct arm_smmu_cmdq_batch *cmds,
-                                   struct arm_smmu_cmdq_ent *cmd)
+                                   struct arm_smmu_cmdq_ent *ent)
 {
-       bool unsupported_cmd = !arm_smmu_cmdq_supports_cmd(cmds->cmdq, cmd);
        bool force_sync = (cmds->num == CMDQ_BATCH_ENTRIES - 1) &&
                          (smmu->options & ARM_SMMU_OPT_CMDQ_FORCE_SYNC);
+       struct arm_smmu_cmd cmd;
+       bool unsupported_cmd;
        int index;
 
+       if (unlikely(arm_smmu_cmdq_build_cmd(cmd.data, ent))) {
+               dev_warn(smmu->dev, "ignoring unknown CMDQ opcode 0x%x\n",
+                        ent->opcode);
+               return;
+       }
+
+       unsupported_cmd = !arm_smmu_cmdq_supports_cmd(cmds->cmdq, &cmd);
        if (force_sync || unsupported_cmd) {
                arm_smmu_cmdq_issue_cmdlist(smmu, cmds->cmdq, cmds->cmds,
                                            cmds->num, true);
-               arm_smmu_cmdq_batch_init(smmu, cmds, cmd);
+               arm_smmu_cmdq_batch_init_cmd(smmu, cmds, &cmd);
        }
 
        if (cmds->num == CMDQ_BATCH_ENTRIES) {
                arm_smmu_cmdq_issue_cmdlist(smmu, cmds->cmdq, cmds->cmds,
                                            cmds->num, false);
-               arm_smmu_cmdq_batch_init(smmu, cmds, cmd);
+               arm_smmu_cmdq_batch_init_cmd(smmu, cmds, &cmd);
        }
 
        index = cmds->num * CMDQ_ENT_DWORDS;
-       if (unlikely(arm_smmu_cmdq_build_cmd(&cmds->cmds[index], cmd))) {
-               dev_warn(smmu->dev, "ignoring unknown CMDQ opcode 0x%x\n",
-                        cmd->opcode);
-               return;
-       }
-
+       memcpy(&cmds->cmds[index], cmd.data, sizeof(cmd.data));
        cmds->num++;
 }
 
index 092179f689e9f1c802335005f641502d7bf587a4..6d73f6b63e64a9bd9d3cc0894a34f2f51ac70aaf 100644 (file)
@@ -641,13 +641,13 @@ struct arm_smmu_cmdq {
        atomic_long_t                   *valid_map;
        atomic_t                        owner_prod;
        atomic_t                        lock;
-       bool                            (*supports_cmd)(struct arm_smmu_cmdq_ent *ent);
+       bool                            (*supports_cmd)(struct arm_smmu_cmd *cmd);
 };
 
 static inline bool arm_smmu_cmdq_supports_cmd(struct arm_smmu_cmdq *cmdq,
-                                             struct arm_smmu_cmdq_ent *ent)
+                                             struct arm_smmu_cmd *cmd)
 {
-       return cmdq->supports_cmd ? cmdq->supports_cmd(ent) : true;
+       return cmdq->supports_cmd ? cmdq->supports_cmd(cmd) : true;
 }
 
 struct arm_smmu_cmdq_batch {
@@ -815,7 +815,7 @@ struct arm_smmu_impl_ops {
        void (*device_remove)(struct arm_smmu_device *smmu);
        int (*init_structures)(struct arm_smmu_device *smmu);
        struct arm_smmu_cmdq *(*get_secondary_cmdq)(
-               struct arm_smmu_device *smmu, struct arm_smmu_cmdq_ent *ent);
+               struct arm_smmu_device *smmu, struct arm_smmu_cmd *cmd);
        /*
         * An implementation should define its own type other than the default
         * IOMMU_HW_INFO_TYPE_ARM_SMMUV3. And it must validate the input @type
index 83f6e9f6c51d6b989c1f1b0d62941c74d3247201..b4d8c1f2fd38784d91b6a2ba890709ae8b32b052 100644 (file)
@@ -367,9 +367,9 @@ static irqreturn_t tegra241_cmdqv_isr(int irq, void *devid)
 
 /* Command Queue Function */
 
-static bool tegra241_guest_vcmdq_supports_cmd(struct arm_smmu_cmdq_ent *ent)
+static bool tegra241_guest_vcmdq_supports_cmd(struct arm_smmu_cmd *cmd)
 {
-       switch (ent->opcode) {
+       switch (FIELD_GET(CMDQ_0_OP, cmd->data[0])) {
        case CMDQ_OP_TLBI_NH_ASID:
        case CMDQ_OP_TLBI_NH_VA:
        case CMDQ_OP_ATC_INV:
@@ -381,7 +381,7 @@ static bool tegra241_guest_vcmdq_supports_cmd(struct arm_smmu_cmdq_ent *ent)
 
 static struct arm_smmu_cmdq *
 tegra241_cmdqv_get_cmdq(struct arm_smmu_device *smmu,
-                       struct arm_smmu_cmdq_ent *ent)
+                       struct arm_smmu_cmd *cmd)
 {
        struct tegra241_cmdqv *cmdqv =
                container_of(smmu, struct tegra241_cmdqv, smmu);
@@ -409,7 +409,7 @@ tegra241_cmdqv_get_cmdq(struct arm_smmu_device *smmu,
                return NULL;
 
        /* Unsupported CMD goes for smmu->cmdq pathway */
-       if (!arm_smmu_cmdq_supports_cmd(&vcmdq->cmdq, ent))
+       if (!arm_smmu_cmdq_supports_cmd(&vcmdq->cmdq, cmd))
                return NULL;
        return &vcmdq->cmdq;
 }