blob: 418a086a3dd14bca30ae501ed352f7f46410d1a6 [file] [log] [blame]
Olav Haugan3c7fb382013-01-02 17:32:25 -08001/* Copyright (c) 2012-2013, The Linux Foundation. All rights reserved.
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -08002 *
3 * This program is free software; you can redistribute it and/or modify
4 * it under the terms of the GNU General Public License version 2 and
5 * only version 2 as published by the Free Software Foundation.
6 *
7 * This program is distributed in the hope that it will be useful,
8 * but WITHOUT ANY WARRANTY; without even the implied warranty of
9 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10 * GNU General Public License for more details.
11 */
12
13#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
14
15#include <linux/kernel.h>
16#include <linux/module.h>
17#include <linux/platform_device.h>
18#include <linux/io.h>
19#include <linux/clk.h>
20#include <linux/iommu.h>
21#include <linux/interrupt.h>
22#include <linux/err.h>
23#include <linux/slab.h>
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -080024#include <linux/of.h>
25#include <linux/of_address.h>
26#include <linux/of_device.h>
27
Olav Haugane6d01ef2013-01-25 16:55:44 -080028#include <mach/iommu_hw-v1.h>
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -080029#include <mach/iommu.h>
Olav Haugan5ebfbc62013-01-07 17:49:10 -080030#include <mach/iommu_perfmon.h>
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -080031
Stepan Moskovchenko880a3182012-10-01 12:35:24 -070032static int msm_iommu_parse_bfb_settings(struct platform_device *pdev,
33 struct msm_iommu_drvdata *drvdata)
34{
35 struct msm_iommu_bfb_settings *bfb_settings;
36 u32 nreg, nval;
Mitchel Humpherysa1830b32013-02-25 18:29:54 -080037 int ret;
Stepan Moskovchenko880a3182012-10-01 12:35:24 -070038
39 /*
40 * It is not valid for a device to have the qcom,iommu-bfb-regs
41 * property but not the qcom,iommu-bfb-data property, and vice versa.
42 */
43 if (!of_get_property(pdev->dev.of_node, "qcom,iommu-bfb-regs", &nreg)) {
44 if (of_get_property(pdev->dev.of_node, "qcom,iommu-bfb-data",
45 &nval))
46 return -EINVAL;
47 return 0;
48 }
49
50 if (!of_get_property(pdev->dev.of_node, "qcom,iommu-bfb-data", &nval))
51 return -EINVAL;
52
53 if (nreg >= sizeof(bfb_settings->regs))
54 return -EINVAL;
55
56 if (nval >= sizeof(bfb_settings->data))
57 return -EINVAL;
58
59 if (nval != nreg)
60 return -EINVAL;
61
62 bfb_settings = devm_kzalloc(&pdev->dev, sizeof(*bfb_settings),
63 GFP_KERNEL);
64 if (!bfb_settings)
65 return -ENOMEM;
66
67 ret = of_property_read_u32_array(pdev->dev.of_node,
68 "qcom,iommu-bfb-regs",
69 bfb_settings->regs,
70 nreg / sizeof(*bfb_settings->regs));
71 if (ret)
72 return ret;
73
74 ret = of_property_read_u32_array(pdev->dev.of_node,
75 "qcom,iommu-bfb-data",
76 bfb_settings->data,
77 nval / sizeof(*bfb_settings->data));
78 if (ret)
79 return ret;
80
81 bfb_settings->length = nreg / sizeof(*bfb_settings->regs);
82
Stepan Moskovchenko880a3182012-10-01 12:35:24 -070083 drvdata->bfb_settings = bfb_settings;
84 return 0;
85}
86
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -080087static int msm_iommu_parse_dt(struct platform_device *pdev,
88 struct msm_iommu_drvdata *drvdata)
89{
90 struct device_node *child;
Stepan Moskovchenko4575bdd2012-06-28 14:59:00 -070091 int ret = 0;
Olav Haugan3c7fb382013-01-02 17:32:25 -080092 struct resource *r;
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -080093
Stephen Boyd55742b72012-08-08 11:40:26 -070094 drvdata->dev = &pdev->dev;
95 msm_iommu_add_drv(drvdata);
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -080096
Stepan Moskovchenko880a3182012-10-01 12:35:24 -070097 ret = msm_iommu_parse_bfb_settings(pdev, drvdata);
98 if (ret)
99 goto fail;
100
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -0800101 for_each_child_of_node(pdev->dev.of_node, child) {
102 drvdata->ncb++;
103 if (!of_platform_device_create(child, NULL, &pdev->dev))
104 pr_err("Failed to create %s device\n", child->name);
105 }
106
Olav Haugan4e315c42013-03-06 10:14:28 -0800107 drvdata->asid = devm_kzalloc(&pdev->dev, drvdata->ncb * sizeof(int),
108 GFP_KERNEL);
109
110 if (!drvdata->asid) {
111 pr_err("Unable to get memory for asid array\n");
112 ret = -ENOMEM;
113 goto fail;
114 }
115
Olav Haugane6c00c32013-01-08 14:11:55 -0800116 ret = of_property_read_string(pdev->dev.of_node, "label",
117 &drvdata->name);
118 if (ret)
119 goto fail;
120
Laura Abbott0d135652012-10-04 12:59:03 -0700121 drvdata->sec_id = -1;
122 of_property_read_u32(pdev->dev.of_node, "qcom,iommu-secure-id",
123 &drvdata->sec_id);
Olav Haugan3c7fb382013-01-02 17:32:25 -0800124
125 r = platform_get_resource_byname(pdev, IORESOURCE_MEM, "clk_base");
126 if (r) {
127 drvdata->clk_reg_virt = devm_ioremap(&pdev->dev, r->start,
128 resource_size(r));
129 if (!drvdata->clk_reg_virt) {
Stepan Moskovchenko61fa3db2012-08-10 17:51:18 +0100130 pr_err("Failed to map resource for iommu clk: %pr\n",
131 r);
Olav Haugan3c7fb382013-01-02 17:32:25 -0800132 ret = -ENOMEM;
133 goto fail;
134 }
135 }
136
Olav Haugancd932192013-01-31 18:30:15 -0800137 drvdata->halt_enabled = of_property_read_bool(pdev->dev.of_node,
138 "qcom,iommu-enable-halt");
139
Laura Abbott0d135652012-10-04 12:59:03 -0700140 return 0;
Stepan Moskovchenko4575bdd2012-06-28 14:59:00 -0700141fail:
142 return ret;
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -0800143}
144
Olav Haugan5ebfbc62013-01-07 17:49:10 -0800145static int msm_iommu_pmon_parse_dt(struct platform_device *pdev,
Olav Haugan0c2d9322013-01-31 18:35:30 -0800146 struct iommu_pmon *pmon_info)
Olav Haugan5ebfbc62013-01-07 17:49:10 -0800147{
148 int ret = 0;
149 int irq = platform_get_irq(pdev, 0);
Olav Haugan0c2d9322013-01-31 18:35:30 -0800150 unsigned int cls_prop_size;
Olav Haugan5ebfbc62013-01-07 17:49:10 -0800151
152 if (irq > 0) {
Olav Haugan0c2d9322013-01-31 18:35:30 -0800153 pmon_info->iommu.evt_irq = platform_get_irq(pdev, 0);
154
155 ret = of_property_read_u32(pdev->dev.of_node,
156 "qcom,iommu-pmu-ngroups",
157 &pmon_info->num_groups);
158 if (ret) {
159 pr_err("Error reading qcom,iommu-pmu-ngroups\n");
160 goto fail;
161 }
162 ret = of_property_read_u32(pdev->dev.of_node,
163 "qcom,iommu-pmu-ncounters",
164 &pmon_info->num_counters);
165 if (ret) {
166 pr_err("Error reading qcom,iommu-pmu-ncounters\n");
167 goto fail;
168 }
169
170 if (!of_get_property(pdev->dev.of_node,
171 "qcom,iommu-pmu-event-classes",
172 &cls_prop_size)) {
173 pr_err("Error reading qcom,iommu-pmu-event-classes\n");
174 return -EINVAL;
175 }
176
177 pmon_info->event_cls_supported =
178 devm_kzalloc(&pdev->dev, cls_prop_size, GFP_KERNEL);
179
180 if (!pmon_info->event_cls_supported) {
181 pr_err("Unable to get memory for event class array\n");
182 return -ENOMEM;
183 }
184
185 pmon_info->nevent_cls_supported = cls_prop_size / sizeof(u32);
186
187 ret = of_property_read_u32_array(pdev->dev.of_node,
188 "qcom,iommu-pmu-event-classes",
189 pmon_info->event_cls_supported,
190 pmon_info->nevent_cls_supported);
191 if (ret) {
192 pr_err("Error reading qcom,iommu-pmu-event-classes\n");
193 return ret;
194 }
Olav Haugan5ebfbc62013-01-07 17:49:10 -0800195 } else {
Olav Haugan0c2d9322013-01-31 18:35:30 -0800196 pmon_info->iommu.evt_irq = -1;
Olav Haugan5ebfbc62013-01-07 17:49:10 -0800197 ret = irq;
198 }
Olav Haugan0c2d9322013-01-31 18:35:30 -0800199
200fail:
Olav Haugan5ebfbc62013-01-07 17:49:10 -0800201 return ret;
202}
203
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -0800204static int __devinit msm_iommu_probe(struct platform_device *pdev)
205{
Olav Haugan0c2d9322013-01-31 18:35:30 -0800206 struct iommu_pmon *pmon_info;
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -0800207 struct msm_iommu_drvdata *drvdata;
208 struct resource *r;
Stepan Moskovchenko17ae71e2012-07-24 19:24:14 -0700209 int ret, needs_alt_core_clk;
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -0800210
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -0800211 drvdata = devm_kzalloc(&pdev->dev, sizeof(*drvdata), GFP_KERNEL);
212 if (!drvdata)
213 return -ENOMEM;
214
Olav Haugan3c7fb382013-01-02 17:32:25 -0800215 r = platform_get_resource_byname(pdev, IORESOURCE_MEM, "iommu_base");
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -0800216 if (!r)
217 return -EINVAL;
218
219 drvdata->base = devm_ioremap(&pdev->dev, r->start, resource_size(r));
220 if (!drvdata->base)
221 return -ENOMEM;
222
Olav Haugan95d24162012-12-05 14:47:47 -0800223 drvdata->glb_base = drvdata->base;
224
Stepan Moskovchenko6751acc2012-06-21 17:36:47 -0700225 drvdata->gdsc = devm_regulator_get(&pdev->dev, "vdd");
226 if (IS_ERR(drvdata->gdsc))
227 return -EINVAL;
228
Olav Haugan2648d972013-01-07 17:32:31 -0800229 drvdata->alt_gdsc = devm_regulator_get(&pdev->dev, "qcom,alt-vdd");
230 if (IS_ERR(drvdata->alt_gdsc))
231 drvdata->alt_gdsc = NULL;
232
Stepan Moskovchenko17ae71e2012-07-24 19:24:14 -0700233 drvdata->pclk = devm_clk_get(&pdev->dev, "iface_clk");
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -0800234 if (IS_ERR(drvdata->pclk))
235 return PTR_ERR(drvdata->pclk);
236
Stepan Moskovchenko17ae71e2012-07-24 19:24:14 -0700237 drvdata->clk = devm_clk_get(&pdev->dev, "core_clk");
238 if (IS_ERR(drvdata->clk))
239 return PTR_ERR(drvdata->clk);
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -0800240
Stepan Moskovchenko17ae71e2012-07-24 19:24:14 -0700241 needs_alt_core_clk = of_property_read_bool(pdev->dev.of_node,
242 "qcom,needs-alt-core-clk");
243 if (needs_alt_core_clk) {
244 drvdata->aclk = devm_clk_get(&pdev->dev, "alt_core_clk");
245 if (IS_ERR(drvdata->aclk))
246 return PTR_ERR(drvdata->aclk);
247 }
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -0800248
Stepan Moskovchenko17ae71e2012-07-24 19:24:14 -0700249 if (clk_get_rate(drvdata->clk) == 0) {
Olav Haugan8d9dfdd2013-02-12 15:00:13 -0800250 ret = clk_round_rate(drvdata->clk, 1000);
Stepan Moskovchenko17ae71e2012-07-24 19:24:14 -0700251 clk_set_rate(drvdata->clk, ret);
252 }
253
254 if (drvdata->aclk && clk_get_rate(drvdata->aclk) == 0) {
Olav Haugan8d9dfdd2013-02-12 15:00:13 -0800255 ret = clk_round_rate(drvdata->aclk, 1000);
Stepan Moskovchenko17ae71e2012-07-24 19:24:14 -0700256 clk_set_rate(drvdata->aclk, ret);
257 }
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -0800258
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -0800259 ret = msm_iommu_parse_dt(pdev, drvdata);
260 if (ret)
Stepan Moskovchenko17ae71e2012-07-24 19:24:14 -0700261 return ret;
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -0800262
Olav Haugan5ebfbc62013-01-07 17:49:10 -0800263 dev_info(&pdev->dev, "device %s mapped at %p, with %d ctx banks\n",
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -0800264 drvdata->name, drvdata->base, drvdata->ncb);
265
266 platform_set_drvdata(pdev, drvdata);
267
Olav Hauganeece7e52013-04-02 10:22:21 -0700268 msm_iommu_sec_set_access_ops(&iommu_access_ops_v1);
269
Olav Haugan5ebfbc62013-01-07 17:49:10 -0800270 pmon_info = msm_iommu_pm_alloc(&pdev->dev);
271 if (pmon_info != NULL) {
272 ret = msm_iommu_pmon_parse_dt(pdev, pmon_info);
273 if (ret) {
274 msm_iommu_pm_free(&pdev->dev);
275 pr_info("%s: pmon not available.\n", drvdata->name);
276 } else {
Olav Haugan0c2d9322013-01-31 18:35:30 -0800277 pmon_info->iommu.base = drvdata->base;
Olav Hauganef69e892013-02-04 13:47:08 -0800278 pmon_info->iommu.ops = &iommu_access_ops_v1;
279 pmon_info->iommu.hw_ops = iommu_pm_get_hw_ops_v1();
Olav Haugan0c2d9322013-01-31 18:35:30 -0800280 pmon_info->iommu.iommu_name = drvdata->name;
Olav Haugan5ebfbc62013-01-07 17:49:10 -0800281 ret = msm_iommu_pm_iommu_register(pmon_info);
282 if (ret) {
283 pr_err("%s iommu register fail\n",
284 drvdata->name);
285 msm_iommu_pm_free(&pdev->dev);
286 } else {
287 pr_debug("%s iommu registered for pmon\n",
Olav Haugan0c2d9322013-01-31 18:35:30 -0800288 pmon_info->iommu.iommu_name);
Olav Haugan5ebfbc62013-01-07 17:49:10 -0800289 }
290 }
291 }
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -0800292 return 0;
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -0800293}
294
295static int __devexit msm_iommu_remove(struct platform_device *pdev)
296{
297 struct msm_iommu_drvdata *drv = NULL;
298
Olav Haugan5ebfbc62013-01-07 17:49:10 -0800299 msm_iommu_pm_iommu_unregister(&pdev->dev);
300 msm_iommu_pm_free(&pdev->dev);
301
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -0800302 drv = platform_get_drvdata(pdev);
303 if (drv) {
Stephen Boyd55742b72012-08-08 11:40:26 -0700304 msm_iommu_remove_drv(drv);
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -0800305 if (drv->clk)
306 clk_put(drv->clk);
307 clk_put(drv->pclk);
308 platform_set_drvdata(pdev, NULL);
309 }
310 return 0;
311}
312
313static int msm_iommu_ctx_parse_dt(struct platform_device *pdev,
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -0800314 struct msm_iommu_ctx_drvdata *ctx_drvdata)
315{
316 struct resource *r, rp;
Sathish Ambleycf045e62012-06-07 12:56:50 -0700317 int irq, ret;
Stepan Moskovchenko4575bdd2012-06-28 14:59:00 -0700318 u32 nsid;
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -0800319
Olav Hauganddd379d2013-01-28 09:07:51 -0800320 ctx_drvdata->secure_context = of_property_read_bool(pdev->dev.of_node,
321 "qcom,secure-context");
322
323 if (!ctx_drvdata->secure_context) {
324 irq = platform_get_irq(pdev, 0);
325 if (irq > 0) {
326 ret = request_threaded_irq(irq, NULL,
327 msm_iommu_fault_handler_v2,
328 IRQF_ONESHOT | IRQF_SHARED,
329 "msm_iommu_nonsecure_irq", pdev);
330 if (ret) {
331 pr_err("Request IRQ %d failed with ret=%d\n",
332 irq, ret);
333 return ret;
334 }
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -0800335 }
336 }
337
338 r = platform_get_resource(pdev, IORESOURCE_MEM, 0);
339 if (!r)
340 return -EINVAL;
341
342 ret = of_address_to_resource(pdev->dev.parent->of_node, 0, &rp);
343 if (ret)
344 return -EINVAL;
345
346 /* Calculate the context bank number using the base addresses. The
347 * first 8 pages belong to the global address space which is followed
348 * by the context banks, hence subtract by 8 to get the context bank
349 * number.
350 */
351 ctx_drvdata->num = ((r->start - rp.start) >> CTX_SHIFT) - 8;
352
Stepan Moskovchenkoce78fc22012-07-11 17:20:27 -0700353 if (of_property_read_string(pdev->dev.of_node, "label",
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -0800354 &ctx_drvdata->name))
355 ctx_drvdata->name = dev_name(&pdev->dev);
356
Stepan Moskovchenko4575bdd2012-06-28 14:59:00 -0700357 if (!of_get_property(pdev->dev.of_node, "qcom,iommu-ctx-sids", &nsid))
358 return -EINVAL;
359
360 if (nsid >= sizeof(ctx_drvdata->sids))
361 return -EINVAL;
362
363 if (of_property_read_u32_array(pdev->dev.of_node, "qcom,iommu-ctx-sids",
364 ctx_drvdata->sids,
365 nsid / sizeof(*ctx_drvdata->sids))) {
366 return -EINVAL;
367 }
368 ctx_drvdata->nsid = nsid;
369
Olav Haugan26ddd432012-12-07 11:39:21 -0800370 ctx_drvdata->asid = -1;
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -0800371 return 0;
372}
373
374static int __devinit msm_iommu_ctx_probe(struct platform_device *pdev)
375{
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -0800376 struct msm_iommu_ctx_drvdata *ctx_drvdata = NULL;
377 int ret;
378
379 if (!pdev->dev.parent)
380 return -EINVAL;
381
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -0800382 ctx_drvdata = devm_kzalloc(&pdev->dev, sizeof(*ctx_drvdata),
383 GFP_KERNEL);
384 if (!ctx_drvdata)
385 return -ENOMEM;
386
387 ctx_drvdata->pdev = pdev;
388 INIT_LIST_HEAD(&ctx_drvdata->attached_elm);
389 platform_set_drvdata(pdev, ctx_drvdata);
390
Sathish Ambleycf045e62012-06-07 12:56:50 -0700391 ret = msm_iommu_ctx_parse_dt(pdev, ctx_drvdata);
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -0800392 if (!ret)
393 dev_info(&pdev->dev, "context %s using bank %d\n",
Stepan Moskovchenkoe14ca5c2012-07-11 12:40:51 -0700394 ctx_drvdata->name, ctx_drvdata->num);
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -0800395
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -0800396 return ret;
397}
398
399static int __devexit msm_iommu_ctx_remove(struct platform_device *pdev)
400{
401 platform_set_drvdata(pdev, NULL);
402 return 0;
403}
404
405static struct of_device_id msm_iommu_match_table[] = {
Olav Haugan0e22c482013-01-28 17:39:36 -0800406 { .compatible = "qcom,msm-smmu-v1", },
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -0800407 {}
408};
409
410static struct platform_driver msm_iommu_driver = {
411 .driver = {
Olav Haugan0e22c482013-01-28 17:39:36 -0800412 .name = "msm_iommu_v1",
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -0800413 .of_match_table = msm_iommu_match_table,
414 },
415 .probe = msm_iommu_probe,
416 .remove = __devexit_p(msm_iommu_remove),
417};
418
419static struct of_device_id msm_iommu_ctx_match_table[] = {
420 { .name = "qcom,iommu-ctx", },
421 {}
422};
423
424static struct platform_driver msm_iommu_ctx_driver = {
425 .driver = {
Olav Haugan0e22c482013-01-28 17:39:36 -0800426 .name = "msm_iommu_ctx_v1",
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -0800427 .of_match_table = msm_iommu_ctx_match_table,
428 },
429 .probe = msm_iommu_ctx_probe,
430 .remove = __devexit_p(msm_iommu_ctx_remove),
431};
432
433static int __init msm_iommu_driver_init(void)
434{
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -0800435 int ret;
436
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -0800437 ret = platform_driver_register(&msm_iommu_driver);
438 if (ret != 0) {
439 pr_err("Failed to register IOMMU driver\n");
440 goto error;
441 }
442
443 ret = platform_driver_register(&msm_iommu_ctx_driver);
444 if (ret != 0) {
445 pr_err("Failed to register IOMMU context driver\n");
446 goto error;
447 }
448
449error:
450 return ret;
451}
452
453static void __exit msm_iommu_driver_exit(void)
454{
455 platform_driver_unregister(&msm_iommu_ctx_driver);
456 platform_driver_unregister(&msm_iommu_driver);
Sathish Ambleyd1b89ed2012-02-07 21:47:47 -0800457}
458
459subsys_initcall(msm_iommu_driver_init);
460module_exit(msm_iommu_driver_exit);
461
462MODULE_LICENSE("GPL v2");