blob: 9392766ad2e2aaf69c0e98049dee4687d921ee74 [file] [log] [blame]
wanwei.jiangd8cfbc12021-12-08 16:32:55 +08001// SPDX-License-Identifier: (GPL-2.0+ OR MIT)
2/*
3 * Copyright (c) 2019 Amlogic, Inc. All rights reserved.
4 */
wanwei.jiang63f85852022-04-20 18:14:26 +08005// #define DEBUG
qinglin.liafe35bd2022-01-18 16:09:02 +08006// #define USE_CMA
7// #define TEST_ACESS
wanwei.jiangd8cfbc12021-12-08 16:32:55 +08008#include <linux/errno.h>
9#include <linux/err.h>
10#include <linux/of.h>
11#include <linux/module.h>
12#include <linux/of_fdt.h>
13#include <linux/libfdt_env.h>
14#include <linux/of_reserved_mem.h>
15#include <linux/io.h>
16#include <linux/platform_device.h>
wanwei.jiang24483b32021-12-08 16:57:18 +080017#include <linux/uaccess.h>
18#ifdef USE_CMA
19#include <linux/dma-map-ops.h>
20#include <linux/cma.h>
21#endif
wanwei.jiangd8cfbc12021-12-08 16:32:55 +080022#include <linux/arm-smccc.h>
Lei Zhang3faeb772022-02-22 09:53:59 +080023#include <linux/memblock.h>
24#include <linux/vmalloc.h>
25#include <linux/slab.h>
26#include <asm/page.h>
wanwei.jiangd8cfbc12021-12-08 16:32:55 +080027
28static void __iomem *sharemem_in_base;
29static void __iomem *sharemem_out_base;
30static long phy_in_base;
31static long phy_out_base;
32static unsigned long secmon_start_virt;
33static unsigned int secmon_size;
34
35#ifdef CONFIG_ARM64
36#define IN_SIZE 0x6000
37#else
38 #define IN_SIZE 0x6000
39#endif
40 #define OUT_SIZE 0x1000
41static DEFINE_MUTEX(sharemem_mutex);
42#define DEV_REGISTERED 1
43#define DEV_UNREGISTED 0
44
45unsigned int sharemem_in_size = IN_SIZE;
46unsigned int sharemem_out_size = OUT_SIZE;
47
48static int secmon_dev_registered = DEV_UNREGISTED;
49static long get_sharemem_info(unsigned int function_id)
50{
51 struct arm_smccc_res res;
52
53 arm_smccc_smc(function_id, 0, 0, 0, 0, 0, 0, 0, &res);
54
55 return res.a0;
56}
57
58static void get_sharemem_size(unsigned int function_id)
59{
60 struct arm_smccc_res res;
61
62 arm_smccc_smc(function_id, 1, 0, 0, 0, 0, 0, 0, &res);
63 if (res.a0 != -1)
64 sharemem_in_size = res.a0;
65
66 arm_smccc_smc(function_id, 2, 0, 0, 0, 0, 0, 0, &res);
67 if (res.a0 != -1)
68 sharemem_out_size = res.a0;
69}
70
71#define RESERVE_MEM_SIZE 0x300000
72
73int within_secmon_region(unsigned long addr)
74{
75 if (!secmon_start_virt)
76 return 0;
77
78 if (addr >= secmon_start_virt &&
79 addr <= (secmon_start_virt + secmon_size))
80 return 1;
81
82 return 0;
83}
wanwei.jiang24483b32021-12-08 16:57:18 +080084EXPORT_SYMBOL(within_secmon_region);
85
86static int get_reserver_base_size(struct platform_device *pdev)
87{
88 struct device_node *mem_region;
89 struct reserved_mem *rmem;
90
91 mem_region = of_parse_phandle(pdev->dev.of_node, "memory-region", 0);
92 if (!mem_region) {
93 dev_warn(&pdev->dev, "no such memory-region\n");
94 return -ENODEV;
95 }
96
97 rmem = of_reserved_mem_lookup(mem_region);
98 if (!rmem) {
99 dev_warn(&pdev->dev, "no such reserved mem of node name %s\n",
100 pdev->dev.of_node->name);
101 return -ENODEV;
102 }
103
104 /* Need to wait for reserved memory to be mapped */
105 //if (!rmem->priv)
106 // return -EPROBE_DEFER;
107
108 if (!rmem->base || !rmem->size) {
109 dev_warn(&pdev->dev, "unexpected reserved memory\n");
110 return -EINVAL;
111 }
112
113 secmon_start_virt = __phys_to_virt(rmem->base);
114
115 //pr_info("secmon_start_virt=0x%016lx, base=0x%010lx, size=0x%010x\n",
116 // secmon_start_virt, rmem->base, rmem->size);
117
118 return 0;
119}
120
121#ifdef TEST_ACESS
122static void test_access_secmon(void)
123{
124 int i, j;
125 int nlines;
126 u32 *p;
127
128 nlines = 16;
129 p = (u32 *)(secmon_start_virt + 0x200000);
130 for (i = 0; i < nlines; i++) {
131 /*
132 * just display low 16 bits of address to keep
133 * each line of the dump < 80 characters
134 */
135 pr_info("%04lx ", (unsigned long)p & 0xffff);
136 for (j = 0; j < 8; j++) {
137 u32 data;
138
139 if (get_kernel_nofault(data, p))
140 pr_cont(" ********");
141 else
142 pr_cont(" %08x", data);
143 ++p;
144 }
145 pr_cont("\n");
146 }
147}
148#endif
wanwei.jiangd8cfbc12021-12-08 16:32:55 +0800149
Lei Zhang3faeb772022-02-22 09:53:59 +0800150static void *ram_vmap(phys_addr_t start, size_t size)
151{
152 struct page **pages;
153 phys_addr_t page_start;
154 unsigned int page_count;
155 unsigned int i;
156 void *vaddr;
157
158 page_start = start - offset_in_page(start);
159 page_count = DIV_ROUND_UP(size + offset_in_page(start), PAGE_SIZE);
160
161 pages = kmalloc_array(page_count, sizeof(struct page *), GFP_KERNEL);
162 if (!pages)
163 return NULL;
164
165 for (i = 0; i < page_count; i++) {
166 phys_addr_t addr = page_start + i * PAGE_SIZE;
167
168 pages[i] = pfn_to_page(addr >> PAGE_SHIFT);
169 }
170 vaddr = vmap(pages, page_count, VM_MAP, PAGE_KERNEL);
171 kfree(pages);
172
173 return vaddr + offset_in_page(start);
174}
175
wanwei.jiangd8cfbc12021-12-08 16:32:55 +0800176static int secmon_probe(struct platform_device *pdev)
177{
178 struct device_node *np = pdev->dev.of_node;
179 unsigned int id;
wanwei.jiang24483b32021-12-08 16:57:18 +0800180 int ret = 0;
181#ifdef USE_CMA
182 struct page *page;
183#endif
wanwei.jiangd8cfbc12021-12-08 16:32:55 +0800184
185 if (!of_property_read_u32(np, "in_base_func", &id))
186 phy_in_base = get_sharemem_info(id);
187
188 if (!of_property_read_u32(np, "out_base_func", &id))
189 phy_out_base = get_sharemem_info(id);
190
191 if (!of_property_read_u32(np, "inout_size_func", &id))
192 get_sharemem_size(id);
193
wanwei.jiang63f85852022-04-20 18:14:26 +0800194 if (!of_property_read_bool (pdev->dev.of_node, "no-memory")) {
195 if (of_property_read_u32(np, "reserve_mem_size", &secmon_size)) {
196 pr_err("can't get reserve_mem_size, use default value\n");
197 secmon_size = RESERVE_MEM_SIZE;
198 } else {
199 pr_info("reserve_mem_size:0x%x\n", secmon_size);
200 }
wanwei.jiang24483b32021-12-08 16:57:18 +0800201
wanwei.jiang63f85852022-04-20 18:14:26 +0800202 get_reserver_base_size(pdev);
wanwei.jiang24483b32021-12-08 16:57:18 +0800203#ifdef USE_CMA
wanwei.jiang63f85852022-04-20 18:14:26 +0800204 ret = of_reserved_mem_device_init(&pdev->dev);
205 if (ret) {
206 pr_err("reserve memory init fail:%d\n", ret);
207 return ret;
208 }
wanwei.jiangd8cfbc12021-12-08 16:32:55 +0800209
wanwei.jiang63f85852022-04-20 18:14:26 +0800210 page = dma_alloc_from_contiguous(&pdev->dev, secmon_size >> PAGE_SHIFT, 0, 0);
211 if (!page) {
212 pr_err("alloc page failed, ret:%p\n", page);
213 return -ENOMEM;
214 }
215 pr_debug("get page:%p, %lx\n", page, page_to_pfn(page));
216 secmon_start_virt = (unsigned long)page_to_virt(page);
wanwei.jiang24483b32021-12-08 16:57:18 +0800217#endif
218
219#ifdef TEST_ACESS
wanwei.jiang63f85852022-04-20 18:14:26 +0800220 test_access_secmon();
wanwei.jiang24483b32021-12-08 16:57:18 +0800221#endif
wanwei.jiang63f85852022-04-20 18:14:26 +0800222 }
223
Lei Zhang3faeb772022-02-22 09:53:59 +0800224 sharemem_in_base = ram_vmap(phy_in_base, sharemem_in_size);
wanwei.jiangd8cfbc12021-12-08 16:32:55 +0800225 if (!sharemem_in_base) {
qinglin.liafe35bd2022-01-18 16:09:02 +0800226 pr_err("secmon share mem in buffer remap fail!\n");
wanwei.jiangd8cfbc12021-12-08 16:32:55 +0800227 return -ENOMEM;
228 }
Lei Zhang3faeb772022-02-22 09:53:59 +0800229 sharemem_out_base = ram_vmap(phy_out_base, sharemem_out_size);
wanwei.jiangd8cfbc12021-12-08 16:32:55 +0800230 if (!sharemem_out_base) {
qinglin.liafe35bd2022-01-18 16:09:02 +0800231 pr_err("secmon share mem out buffer remap fail!\n");
wanwei.jiangd8cfbc12021-12-08 16:32:55 +0800232 return -ENOMEM;
233 }
234 secmon_dev_registered = DEV_REGISTERED;
235 pr_info("share in base: 0x%lx, share out base: 0x%lx\n",
236 (long)sharemem_in_base, (long)sharemem_out_base);
237 pr_info("phy_in_base: 0x%lx, phy_out_base: 0x%lx\n",
238 phy_in_base, phy_out_base);
239
240 return ret;
241}
242
wanwei.jiang24483b32021-12-08 16:57:18 +0800243#ifdef USE_CMA
244void __init secmon_clear_cma_mmu(void)
wanwei.jiangd8cfbc12021-12-08 16:32:55 +0800245{
246 struct device_node *np;
247 unsigned int clear[2] = {};
248
249 np = of_find_node_by_name(NULL, "secmon");
250 if (!np)
251 return;
252
253 if (of_property_read_u32_array(np, "clear_range", clear, 2))
254 pr_info("can't fine clear_range\n");
255 else
256 pr_info("clear_range:%x %x\n", clear[0], clear[1]);
257
258 if (clear[0]) {
259 struct page *page = phys_to_page(clear[0]);
260 int cnt = clear[1] / PAGE_SIZE;
261
wanwei.jiang24483b32021-12-08 16:57:18 +0800262 cma_mmu_op(page, cnt, 0); //cma_mmu_op() implemented in kernel
wanwei.jiangd8cfbc12021-12-08 16:32:55 +0800263 }
wanwei.jiang24483b32021-12-08 16:57:18 +0800264}
265#endif
wanwei.jiangd8cfbc12021-12-08 16:32:55 +0800266
267static const struct of_device_id secmon_dt_match[] = {
268 { .compatible = "amlogic, secmon" },
269 { /* sentinel */ },
270};
271
272static struct platform_driver secmon_platform_driver = {
273 .probe = secmon_probe,
274 .driver = {
275 .owner = THIS_MODULE,
276 .name = "secmon",
277 .of_match_table = secmon_dt_match,
278 },
279};
280
281int __init meson_secmon_init(void)
282{
283 int ret;
284
285 ret = platform_driver_register(&secmon_platform_driver);
286 WARN((secmon_dev_registered != DEV_REGISTERED),
wanwei.jiang24483b32021-12-08 16:57:18 +0800287 "ERROR: secmon device must be enable!!!\n");
wanwei.jiangd8cfbc12021-12-08 16:32:55 +0800288 return ret;
289}
290
291#ifdef MODULE
292module_init(meson_secmon_init);
293MODULE_LICENSE("GPL v2");
294#else
295subsys_initcall(meson_secmon_init);
296#endif
297
298void meson_sm_mutex_lock(void)
299{
300 mutex_lock(&sharemem_mutex);
301}
302EXPORT_SYMBOL(meson_sm_mutex_lock);
303
304void meson_sm_mutex_unlock(void)
305{
306 mutex_unlock(&sharemem_mutex);
307}
308EXPORT_SYMBOL(meson_sm_mutex_unlock);
309
310void __iomem *get_meson_sm_input_base(void)
311{
312 return sharemem_in_base;
313}
314EXPORT_SYMBOL(get_meson_sm_input_base);
wanwei.jiangdd140762022-04-27 20:56:00 +0800315
wanwei.jiangd8cfbc12021-12-08 16:32:55 +0800316void __iomem *get_meson_sm_output_base(void)
317{
318 return sharemem_out_base;
319}
320EXPORT_SYMBOL(get_meson_sm_output_base);
321
322long get_secmon_phy_input_base(void)
323{
324 return phy_in_base;
325}
326EXPORT_SYMBOL(get_secmon_phy_input_base);
327
328long get_secmon_phy_output_base(void)
329{
330 return phy_out_base;
331}
wanwei.jiangdd140762022-04-27 20:56:00 +0800332EXPORT_SYMBOL(get_secmon_phy_output_base);
wanwei.jiangd8cfbc12021-12-08 16:32:55 +0800333
334unsigned int get_secmon_sharemem_in_size(void)
335{
336 return sharemem_in_size;
337}
338EXPORT_SYMBOL(get_secmon_sharemem_in_size);
wanwei.jiangdd140762022-04-27 20:56:00 +0800339
wanwei.jiangd8cfbc12021-12-08 16:32:55 +0800340unsigned int get_secmon_sharemem_out_size(void)
341{
342 return sharemem_out_size;
343}
344EXPORT_SYMBOL(get_secmon_sharemem_out_size);
345