blob: a362a7e3dc5f770e947a4fde4c83cfa6b6ef23e8 [file] [log] [blame]
Matt Wagantallb3fe8992011-12-07 19:26:55 -08001/*
2 * Copyright (c) 2012, Code Aurora Forum. All rights reserved.
3 *
4 * This program is free software; you can redistribute it and/or modify
5 * it under the terms of the GNU General Public License version 2 and
6 * only version 2 as published by the Free Software Foundation.
7 *
8 * This program is distributed in the hope that it will be useful,
9 * but WITHOUT ANY WARRANTY; without even the implied warranty of
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 * GNU General Public License for more details.
12 */
13
14#include <linux/init.h>
15#include <linux/module.h>
16#include <linux/platform_device.h>
17#include <linux/io.h>
Matt Wagantallb7747992012-05-11 19:37:51 -070018#include <linux/iopoll.h>
Matt Wagantallb3fe8992011-12-07 19:26:55 -080019#include <linux/elf.h>
20#include <linux/err.h>
21#include <linux/of.h>
22#include <linux/clk.h>
23
Matt Wagantalld41ce772012-05-10 23:16:41 -070024#include <mach/clk.h>
25
Matt Wagantallb3fe8992011-12-07 19:26:55 -080026#include "peripheral-loader.h"
27#include "pil-q6v5.h"
28
Matt Wagantallb7747992012-05-11 19:37:51 -070029/* QDSP6SS Register Offsets */
Matt Wagantallb3fe8992011-12-07 19:26:55 -080030#define QDSP6SS_RESET 0x014
31#define QDSP6SS_GFMUX_CTL 0x020
32#define QDSP6SS_PWR_CTL 0x030
33
Matt Wagantallb7747992012-05-11 19:37:51 -070034/* AXI Halt Register Offsets */
35#define AXI_HALTREQ 0x0
36#define AXI_HALTACK 0x4
37#define AXI_IDLE 0x8
38
39#define HALT_ACK_TIMEOUT_US 100000
40
Matt Wagantallb3fe8992011-12-07 19:26:55 -080041/* QDSP6SS_RESET */
42#define Q6SS_CORE_ARES BIT(1)
43#define Q6SS_ETM_ISDB_ARES BIT(3)
44#define Q6SS_STOP_CORE BIT(4)
45
46/* QDSP6SS_GFMUX_CTL */
47#define Q6SS_CLK_ENA BIT(1)
48
49/* QDSP6SS_PWR_CTL */
50#define Q6SS_L2DATA_SLP_NRET_N BIT(0)
51#define Q6SS_L2TAG_SLP_NRET_N BIT(16)
52#define Q6SS_ETB_SLP_NRET_N BIT(17)
53#define Q6SS_L2DATA_STBY_N BIT(18)
54#define Q6SS_SLP_RET_N BIT(19)
55#define Q6SS_CLAMP_IO BIT(20)
56#define QDSS_BHS_ON BIT(21)
57
58int pil_q6v5_make_proxy_votes(struct pil_desc *pil)
59{
60 int ret;
61 struct q6v5_data *drv = dev_get_drvdata(pil->dev);
62
63 ret = clk_prepare_enable(drv->xo);
64 if (ret) {
65 dev_err(pil->dev, "Failed to enable XO\n");
66 return ret;
67 }
68 return 0;
69}
70EXPORT_SYMBOL(pil_q6v5_make_proxy_votes);
71
72void pil_q6v5_remove_proxy_votes(struct pil_desc *pil)
73{
74 struct q6v5_data *drv = dev_get_drvdata(pil->dev);
75 clk_disable_unprepare(drv->xo);
76}
77EXPORT_SYMBOL(pil_q6v5_remove_proxy_votes);
78
Matt Wagantallb7747992012-05-11 19:37:51 -070079void pil_q6v5_halt_axi_port(struct pil_desc *pil, void __iomem *halt_base)
80{
81 int ret;
82 u32 status;
83
84 /* Assert halt request */
85 writel_relaxed(1, halt_base + AXI_HALTREQ);
86
87 /* Wait for halt */
88 ret = readl_poll_timeout(halt_base + AXI_HALTACK,
89 status, status != 0, 50, HALT_ACK_TIMEOUT_US);
90 if (ret)
91 dev_warn(pil->dev, "Port %p halt timeout\n", halt_base);
92 else if (!readl_relaxed(halt_base + AXI_IDLE))
93 dev_warn(pil->dev, "Port %p halt failed\n", halt_base);
94
95 /* Clear halt request (port will remain halted until reset) */
96 writel_relaxed(0, halt_base + AXI_HALTREQ);
97}
98EXPORT_SYMBOL(pil_q6v5_halt_axi_port);
99
Matt Wagantallb3fe8992011-12-07 19:26:55 -0800100int pil_q6v5_init_image(struct pil_desc *pil, const u8 *metadata,
101 size_t size)
102{
103 const struct elf32_hdr *ehdr = (struct elf32_hdr *)metadata;
104 struct q6v5_data *drv = dev_get_drvdata(pil->dev);
105 drv->start_addr = ehdr->e_entry;
106 return 0;
107}
108EXPORT_SYMBOL(pil_q6v5_init_image);
109
Matt Wagantalld41ce772012-05-10 23:16:41 -0700110int pil_q6v5_enable_clks(struct pil_desc *pil)
111{
112 struct q6v5_data *drv = dev_get_drvdata(pil->dev);
113 int ret;
114
115 ret = clk_reset(drv->core_clk, CLK_RESET_DEASSERT);
116 if (ret)
117 goto err_reset;
118 ret = clk_prepare_enable(drv->core_clk);
119 if (ret)
120 goto err_core_clk;
121 ret = clk_prepare_enable(drv->bus_clk);
122 if (ret)
123 goto err_bus_clk;
Matt Wagantall4e2599e2012-03-21 22:31:35 -0700124 if (drv->mem_clk) {
125 ret = clk_prepare_enable(drv->mem_clk);
126 if (ret)
127 goto err_mem_clk;
128 }
Matt Wagantalld41ce772012-05-10 23:16:41 -0700129
130 return 0;
131
Matt Wagantall4e2599e2012-03-21 22:31:35 -0700132err_mem_clk:
133 clk_disable_unprepare(drv->bus_clk);
Matt Wagantalld41ce772012-05-10 23:16:41 -0700134err_bus_clk:
135 clk_disable_unprepare(drv->core_clk);
136err_core_clk:
137 clk_reset(drv->core_clk, CLK_RESET_ASSERT);
138err_reset:
139 return ret;
140}
141EXPORT_SYMBOL(pil_q6v5_enable_clks);
142
143void pil_q6v5_disable_clks(struct pil_desc *pil)
144{
145 struct q6v5_data *drv = dev_get_drvdata(pil->dev);
146
147 clk_disable_unprepare(drv->bus_clk);
148 clk_disable_unprepare(drv->core_clk);
Matt Wagantall4e2599e2012-03-21 22:31:35 -0700149 clk_disable_unprepare(drv->mem_clk);
Matt Wagantalld41ce772012-05-10 23:16:41 -0700150 clk_reset(drv->core_clk, CLK_RESET_ASSERT);
151}
152EXPORT_SYMBOL(pil_q6v5_disable_clks);
153
Matt Wagantallb3fe8992011-12-07 19:26:55 -0800154void pil_q6v5_shutdown(struct pil_desc *pil)
155{
156 u32 val;
157 struct q6v5_data *drv = dev_get_drvdata(pil->dev);
158
159 /* Turn off core clock */
160 val = readl_relaxed(drv->reg_base + QDSP6SS_GFMUX_CTL);
161 val &= ~Q6SS_CLK_ENA;
162 writel_relaxed(val, drv->reg_base + QDSP6SS_GFMUX_CTL);
163
164 /* Clamp IO */
165 val = readl_relaxed(drv->reg_base + QDSP6SS_PWR_CTL);
166 val |= Q6SS_CLAMP_IO;
167 writel_relaxed(val, drv->reg_base + QDSP6SS_PWR_CTL);
168
169 /* Turn off Q6 memories */
170 val &= ~(Q6SS_L2DATA_SLP_NRET_N | Q6SS_SLP_RET_N |
171 Q6SS_L2TAG_SLP_NRET_N | Q6SS_ETB_SLP_NRET_N |
172 Q6SS_L2DATA_STBY_N);
173 writel_relaxed(Q6SS_CLAMP_IO, drv->reg_base + QDSP6SS_PWR_CTL);
174
175 /* Assert Q6 resets */
176 val = readl_relaxed(drv->reg_base + QDSP6SS_RESET);
177 val = (Q6SS_CORE_ARES | Q6SS_ETM_ISDB_ARES);
178 writel_relaxed(val, drv->reg_base + QDSP6SS_RESET);
179
180 /* Kill power at block headswitch (affects LPASS only) */
181 val = readl_relaxed(drv->reg_base + QDSP6SS_PWR_CTL);
182 val &= ~QDSS_BHS_ON;
183 writel_relaxed(val, drv->reg_base + QDSP6SS_PWR_CTL);
184}
185EXPORT_SYMBOL(pil_q6v5_shutdown);
186
187int pil_q6v5_reset(struct pil_desc *pil)
188{
189 struct q6v5_data *drv = dev_get_drvdata(pil->dev);
190 u32 val;
191
192 /* Assert resets, stop core */
193 val = readl_relaxed(drv->reg_base + QDSP6SS_RESET);
194 val |= (Q6SS_CORE_ARES | Q6SS_ETM_ISDB_ARES | Q6SS_STOP_CORE);
195 writel_relaxed(val, drv->reg_base + QDSP6SS_RESET);
196
197 /* Enable power block headswitch (only affects LPASS) */
198 val = readl_relaxed(drv->reg_base + QDSP6SS_PWR_CTL);
199 val |= QDSS_BHS_ON;
200 writel_relaxed(val, drv->reg_base + QDSP6SS_PWR_CTL);
201
202 /* Turn on memories */
203 val = readl_relaxed(drv->reg_base + QDSP6SS_PWR_CTL);
204 val |= Q6SS_L2DATA_SLP_NRET_N | Q6SS_SLP_RET_N |
205 Q6SS_L2TAG_SLP_NRET_N | Q6SS_ETB_SLP_NRET_N |
206 Q6SS_L2DATA_STBY_N;
207 writel_relaxed(val, drv->reg_base + QDSP6SS_PWR_CTL);
208
209 /* Remove IO clamp */
210 val &= ~Q6SS_CLAMP_IO;
211 writel_relaxed(val, drv->reg_base + QDSP6SS_PWR_CTL);
212
213 /* Bring core out of reset */
214 val = Q6SS_STOP_CORE;
215 writel_relaxed(val, drv->reg_base + QDSP6SS_RESET);
216
217 /* Turn on core clock */
218 val = readl_relaxed(drv->reg_base + QDSP6SS_GFMUX_CTL);
219 val |= Q6SS_CLK_ENA;
220 writel_relaxed(val, drv->reg_base + QDSP6SS_GFMUX_CTL);
221
222 /* Start core execution */
223 val = readl_relaxed(drv->reg_base + QDSP6SS_RESET);
224 val &= ~Q6SS_STOP_CORE;
225 writel_relaxed(val, drv->reg_base + QDSP6SS_RESET);
226
227 return 0;
228}
229EXPORT_SYMBOL(pil_q6v5_reset);
230
231struct pil_desc __devinit *pil_q6v5_init(struct platform_device *pdev)
232{
233 struct q6v5_data *drv;
234 struct resource *res;
235 struct pil_desc *desc;
236 int ret;
237
238 drv = devm_kzalloc(&pdev->dev, sizeof(*drv), GFP_KERNEL);
239 if (!drv)
240 return ERR_PTR(-ENOMEM);
241 platform_set_drvdata(pdev, drv);
242
243 res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
244 if (!res)
245 return ERR_PTR(-EINVAL);
246 drv->reg_base = devm_ioremap(&pdev->dev, res->start,
247 resource_size(res));
248 if (!drv->reg_base)
249 return ERR_PTR(-ENOMEM);
Matt Wagantallb7747992012-05-11 19:37:51 -0700250 res = platform_get_resource(pdev, IORESOURCE_MEM, 1);
251 drv->axi_halt_base = devm_ioremap(&pdev->dev, res->start,
252 resource_size(res));
253 if (!drv->axi_halt_base)
254 return ERR_PTR(-ENOMEM);
Matt Wagantallb3fe8992011-12-07 19:26:55 -0800255
Matt Wagantallb3fe8992011-12-07 19:26:55 -0800256 desc = devm_kzalloc(&pdev->dev, sizeof(*desc), GFP_KERNEL);
257 if (!desc)
258 return ERR_PTR(-ENOMEM);
259
260 ret = of_property_read_string(pdev->dev.of_node, "qcom,firmware-name",
261 &desc->name);
262 if (ret)
263 return ERR_PTR(ret);
264
265 drv->xo = devm_clk_get(&pdev->dev, "xo");
266 if (IS_ERR(drv->xo))
267 return ERR_CAST(drv->xo);
268
Matt Wagantalld41ce772012-05-10 23:16:41 -0700269 drv->bus_clk = devm_clk_get(&pdev->dev, "bus_clk");
270 if (IS_ERR(drv->bus_clk))
271 return ERR_CAST(drv->bus_clk);
272
273 drv->core_clk = devm_clk_get(&pdev->dev, "core_clk");
274 if (IS_ERR(drv->core_clk))
275 return ERR_CAST(drv->core_clk);
276
Matt Wagantallb3fe8992011-12-07 19:26:55 -0800277 desc->dev = &pdev->dev;
278
279 return desc;
280}
281EXPORT_SYMBOL(pil_q6v5_init);