1 | // SPDX-License-Identifier: GPL-2.0-only |
2 | /* Copyright (c) 2015,2019 The Linux Foundation. All rights reserved. |
3 | */ |
4 | |
5 | #include <linux/cleanup.h> |
6 | #include <linux/io.h> |
7 | #include <linux/errno.h> |
8 | #include <linux/delay.h> |
9 | #include <linux/mutex.h> |
10 | #include <linux/slab.h> |
11 | #include <linux/types.h> |
12 | #include <linux/firmware/qcom/qcom_scm.h> |
13 | #include <linux/firmware/qcom/qcom_tzmem.h> |
14 | #include <linux/arm-smccc.h> |
15 | #include <linux/dma-mapping.h> |
16 | |
17 | #include "qcom_scm.h" |
18 | |
19 | /** |
20 | * struct arm_smccc_args |
21 | * @args: The array of values used in registers in smc instruction |
22 | */ |
23 | struct arm_smccc_args { |
24 | unsigned long args[8]; |
25 | }; |
26 | |
27 | static DEFINE_MUTEX(qcom_scm_lock); |
28 | |
29 | #define QCOM_SCM_EBUSY_WAIT_MS 30 |
30 | #define QCOM_SCM_EBUSY_MAX_RETRY 20 |
31 | |
32 | #define SCM_SMC_N_REG_ARGS 4 |
33 | #define SCM_SMC_FIRST_EXT_IDX (SCM_SMC_N_REG_ARGS - 1) |
34 | #define SCM_SMC_N_EXT_ARGS (MAX_QCOM_SCM_ARGS - SCM_SMC_N_REG_ARGS + 1) |
35 | #define SCM_SMC_FIRST_REG_IDX 2 |
36 | #define SCM_SMC_LAST_REG_IDX (SCM_SMC_FIRST_REG_IDX + SCM_SMC_N_REG_ARGS - 1) |
37 | |
38 | static void __scm_smc_do_quirk(const struct arm_smccc_args *smc, |
39 | struct arm_smccc_res *res) |
40 | { |
41 | unsigned long a0 = smc->args[0]; |
42 | struct arm_smccc_quirk quirk = { .id = ARM_SMCCC_QUIRK_QCOM_A6 }; |
43 | |
44 | quirk.state.a6 = 0; |
45 | |
46 | do { |
47 | arm_smccc_smc_quirk(a0, smc->args[1], smc->args[2], |
48 | smc->args[3], smc->args[4], smc->args[5], |
49 | quirk.state.a6, smc->args[7], res, &quirk); |
50 | |
51 | if (res->a0 == QCOM_SCM_INTERRUPTED) |
52 | a0 = res->a0; |
53 | |
54 | } while (res->a0 == QCOM_SCM_INTERRUPTED); |
55 | } |
56 | |
57 | static void fill_wq_resume_args(struct arm_smccc_args *resume, u32 smc_call_ctx) |
58 | { |
59 | memset(resume->args, 0, sizeof(resume->args[0]) * ARRAY_SIZE(resume->args)); |
60 | |
61 | resume->args[0] = ARM_SMCCC_CALL_VAL(ARM_SMCCC_STD_CALL, |
62 | ARM_SMCCC_SMC_64, ARM_SMCCC_OWNER_SIP, |
63 | SCM_SMC_FNID(QCOM_SCM_SVC_WAITQ, QCOM_SCM_WAITQ_RESUME)); |
64 | |
65 | resume->args[1] = QCOM_SCM_ARGS(1); |
66 | |
67 | resume->args[2] = smc_call_ctx; |
68 | } |
69 | |
70 | int scm_get_wq_ctx(u32 *wq_ctx, u32 *flags, u32 *more_pending) |
71 | { |
72 | int ret; |
73 | struct arm_smccc_res get_wq_res; |
74 | struct arm_smccc_args get_wq_ctx = {0}; |
75 | |
76 | get_wq_ctx.args[0] = ARM_SMCCC_CALL_VAL(ARM_SMCCC_FAST_CALL, |
77 | ARM_SMCCC_SMC_64, ARM_SMCCC_OWNER_SIP, |
78 | SCM_SMC_FNID(QCOM_SCM_SVC_WAITQ, QCOM_SCM_WAITQ_GET_WQ_CTX)); |
79 | |
80 | /* Guaranteed to return only success or error, no WAITQ_* */ |
81 | __scm_smc_do_quirk(smc: &get_wq_ctx, res: &get_wq_res); |
82 | ret = get_wq_res.a0; |
83 | if (ret) |
84 | return ret; |
85 | |
86 | *wq_ctx = get_wq_res.a1; |
87 | *flags = get_wq_res.a2; |
88 | *more_pending = get_wq_res.a3; |
89 | |
90 | return 0; |
91 | } |
92 | |
93 | static int __scm_smc_do_quirk_handle_waitq(struct device *dev, struct arm_smccc_args *waitq, |
94 | struct arm_smccc_res *res) |
95 | { |
96 | int ret; |
97 | u32 wq_ctx, smc_call_ctx; |
98 | struct arm_smccc_args resume; |
99 | struct arm_smccc_args *smc = waitq; |
100 | |
101 | do { |
102 | __scm_smc_do_quirk(smc, res); |
103 | |
104 | if (res->a0 == QCOM_SCM_WAITQ_SLEEP) { |
105 | wq_ctx = res->a1; |
106 | smc_call_ctx = res->a2; |
107 | |
108 | ret = qcom_scm_wait_for_wq_completion(wq_ctx); |
109 | if (ret) |
110 | return ret; |
111 | |
112 | fill_wq_resume_args(resume: &resume, smc_call_ctx); |
113 | smc = &resume; |
114 | } |
115 | } while (res->a0 == QCOM_SCM_WAITQ_SLEEP); |
116 | |
117 | return 0; |
118 | } |
119 | |
120 | static int __scm_smc_do(struct device *dev, struct arm_smccc_args *smc, |
121 | struct arm_smccc_res *res, bool atomic) |
122 | { |
123 | int ret, retry_count = 0; |
124 | |
125 | if (atomic) { |
126 | __scm_smc_do_quirk(smc, res); |
127 | return 0; |
128 | } |
129 | |
130 | do { |
131 | mutex_lock(&qcom_scm_lock); |
132 | |
133 | ret = __scm_smc_do_quirk_handle_waitq(dev, waitq: smc, res); |
134 | |
135 | mutex_unlock(lock: &qcom_scm_lock); |
136 | |
137 | if (ret) |
138 | return ret; |
139 | |
140 | if (res->a0 == QCOM_SCM_V2_EBUSY) { |
141 | if (retry_count++ > QCOM_SCM_EBUSY_MAX_RETRY) |
142 | break; |
143 | msleep(QCOM_SCM_EBUSY_WAIT_MS); |
144 | } |
145 | } while (res->a0 == QCOM_SCM_V2_EBUSY); |
146 | |
147 | return 0; |
148 | } |
149 | |
150 | |
151 | int __scm_smc_call(struct device *dev, const struct qcom_scm_desc *desc, |
152 | enum qcom_scm_convention qcom_convention, |
153 | struct qcom_scm_res *res, bool atomic) |
154 | { |
155 | int arglen = desc->arginfo & 0xf; |
156 | int i, ret; |
157 | void *args_virt __free(qcom_tzmem) = NULL; |
158 | gfp_t flag = atomic ? GFP_ATOMIC : GFP_KERNEL; |
159 | u32 smccc_call_type = atomic ? ARM_SMCCC_FAST_CALL : ARM_SMCCC_STD_CALL; |
160 | u32 qcom_smccc_convention = (qcom_convention == SMC_CONVENTION_ARM_32) ? |
161 | ARM_SMCCC_SMC_32 : ARM_SMCCC_SMC_64; |
162 | struct arm_smccc_res smc_res; |
163 | struct arm_smccc_args smc = {0}; |
164 | |
165 | smc.args[0] = ARM_SMCCC_CALL_VAL( |
166 | smccc_call_type, |
167 | qcom_smccc_convention, |
168 | desc->owner, |
169 | SCM_SMC_FNID(desc->svc, desc->cmd)); |
170 | smc.args[1] = desc->arginfo; |
171 | for (i = 0; i < SCM_SMC_N_REG_ARGS; i++) |
172 | smc.args[i + SCM_SMC_FIRST_REG_IDX] = desc->args[i]; |
173 | |
174 | if (unlikely(arglen > SCM_SMC_N_REG_ARGS)) { |
175 | struct qcom_tzmem_pool *mempool = qcom_scm_get_tzmem_pool(); |
176 | |
177 | if (!mempool) |
178 | return -EINVAL; |
179 | |
180 | args_virt = qcom_tzmem_alloc(pool: mempool, |
181 | SCM_SMC_N_EXT_ARGS * sizeof(u64), |
182 | gfp: flag); |
183 | if (!args_virt) |
184 | return -ENOMEM; |
185 | |
186 | if (qcom_smccc_convention == ARM_SMCCC_SMC_32) { |
187 | __le32 *args = args_virt; |
188 | |
189 | for (i = 0; i < SCM_SMC_N_EXT_ARGS; i++) |
190 | args[i] = cpu_to_le32(desc->args[i + |
191 | SCM_SMC_FIRST_EXT_IDX]); |
192 | } else { |
193 | __le64 *args = args_virt; |
194 | |
195 | for (i = 0; i < SCM_SMC_N_EXT_ARGS; i++) |
196 | args[i] = cpu_to_le64(desc->args[i + |
197 | SCM_SMC_FIRST_EXT_IDX]); |
198 | } |
199 | |
200 | smc.args[SCM_SMC_LAST_REG_IDX] = qcom_tzmem_to_phys(ptr: args_virt); |
201 | } |
202 | |
203 | ret = __scm_smc_do(dev, smc: &smc, res: &smc_res, atomic); |
204 | if (ret) |
205 | return ret; |
206 | |
207 | if (res) { |
208 | res->result[0] = smc_res.a1; |
209 | res->result[1] = smc_res.a2; |
210 | res->result[2] = smc_res.a3; |
211 | } |
212 | |
213 | return (long)smc_res.a0 ? qcom_scm_remap_error(err: smc_res.a0) : 0; |
214 | |
215 | } |
216 | |