blob: d6ad2aa39e5ec50fdf30acafde640e012d408f3d [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
Matt Wagantallb76f1902012-08-09 14:51:28 -070033#define QDSP6SS_CGC_OVERRIDE 0x034
Matt Wagantallb3fe8992011-12-07 19:26:55 -080034
Matt Wagantallb7747992012-05-11 19:37:51 -070035/* AXI Halt Register Offsets */
36#define AXI_HALTREQ 0x0
37#define AXI_HALTACK 0x4
38#define AXI_IDLE 0x8
39
40#define HALT_ACK_TIMEOUT_US 100000
41
Matt Wagantallb3fe8992011-12-07 19:26:55 -080042/* QDSP6SS_RESET */
Matt Wagantall11c07e22012-08-09 16:14:07 -070043#define Q6SS_STOP_CORE BIT(0)
Matt Wagantallb3fe8992011-12-07 19:26:55 -080044#define Q6SS_CORE_ARES BIT(1)
Matt Wagantall11c07e22012-08-09 16:14:07 -070045#define Q6SS_BUS_ARES_ENA BIT(2)
Matt Wagantallb3fe8992011-12-07 19:26:55 -080046
47/* QDSP6SS_GFMUX_CTL */
48#define Q6SS_CLK_ENA BIT(1)
49
50/* QDSP6SS_PWR_CTL */
Matt Wagantall11c07e22012-08-09 16:14:07 -070051#define Q6SS_L2DATA_SLP_NRET_N (BIT(0)|BIT(1)|BIT(2))
Matt Wagantallb3fe8992011-12-07 19:26:55 -080052#define Q6SS_L2TAG_SLP_NRET_N BIT(16)
53#define Q6SS_ETB_SLP_NRET_N BIT(17)
54#define Q6SS_L2DATA_STBY_N BIT(18)
55#define Q6SS_SLP_RET_N BIT(19)
56#define Q6SS_CLAMP_IO BIT(20)
57#define QDSS_BHS_ON BIT(21)
58
Matt Wagantallb76f1902012-08-09 14:51:28 -070059/* QDSP6SS_CGC_OVERRIDE */
60#define Q6SS_CORE_CLK_EN BIT(0)
61#define Q6SS_CORE_RCLK_EN BIT(1)
62
Matt Wagantallb3fe8992011-12-07 19:26:55 -080063int pil_q6v5_make_proxy_votes(struct pil_desc *pil)
64{
65 int ret;
66 struct q6v5_data *drv = dev_get_drvdata(pil->dev);
67
68 ret = clk_prepare_enable(drv->xo);
69 if (ret) {
70 dev_err(pil->dev, "Failed to enable XO\n");
71 return ret;
72 }
73 return 0;
74}
75EXPORT_SYMBOL(pil_q6v5_make_proxy_votes);
76
77void pil_q6v5_remove_proxy_votes(struct pil_desc *pil)
78{
79 struct q6v5_data *drv = dev_get_drvdata(pil->dev);
80 clk_disable_unprepare(drv->xo);
81}
82EXPORT_SYMBOL(pil_q6v5_remove_proxy_votes);
83
Matt Wagantallb7747992012-05-11 19:37:51 -070084void pil_q6v5_halt_axi_port(struct pil_desc *pil, void __iomem *halt_base)
85{
86 int ret;
87 u32 status;
88
89 /* Assert halt request */
90 writel_relaxed(1, halt_base + AXI_HALTREQ);
91
92 /* Wait for halt */
93 ret = readl_poll_timeout(halt_base + AXI_HALTACK,
94 status, status != 0, 50, HALT_ACK_TIMEOUT_US);
95 if (ret)
96 dev_warn(pil->dev, "Port %p halt timeout\n", halt_base);
97 else if (!readl_relaxed(halt_base + AXI_IDLE))
98 dev_warn(pil->dev, "Port %p halt failed\n", halt_base);
99
100 /* Clear halt request (port will remain halted until reset) */
101 writel_relaxed(0, halt_base + AXI_HALTREQ);
102}
103EXPORT_SYMBOL(pil_q6v5_halt_axi_port);
104
Matt Wagantallb3fe8992011-12-07 19:26:55 -0800105int pil_q6v5_init_image(struct pil_desc *pil, const u8 *metadata,
106 size_t size)
107{
108 const struct elf32_hdr *ehdr = (struct elf32_hdr *)metadata;
109 struct q6v5_data *drv = dev_get_drvdata(pil->dev);
110 drv->start_addr = ehdr->e_entry;
111 return 0;
112}
113EXPORT_SYMBOL(pil_q6v5_init_image);
114
Matt Wagantalld41ce772012-05-10 23:16:41 -0700115int pil_q6v5_enable_clks(struct pil_desc *pil)
116{
117 struct q6v5_data *drv = dev_get_drvdata(pil->dev);
118 int ret;
119
120 ret = clk_reset(drv->core_clk, CLK_RESET_DEASSERT);
121 if (ret)
122 goto err_reset;
123 ret = clk_prepare_enable(drv->core_clk);
124 if (ret)
125 goto err_core_clk;
126 ret = clk_prepare_enable(drv->bus_clk);
127 if (ret)
128 goto err_bus_clk;
Matt Wagantall4e2599e2012-03-21 22:31:35 -0700129 if (drv->mem_clk) {
130 ret = clk_prepare_enable(drv->mem_clk);
131 if (ret)
132 goto err_mem_clk;
133 }
Matt Wagantalld41ce772012-05-10 23:16:41 -0700134
135 return 0;
136
Matt Wagantall4e2599e2012-03-21 22:31:35 -0700137err_mem_clk:
138 clk_disable_unprepare(drv->bus_clk);
Matt Wagantalld41ce772012-05-10 23:16:41 -0700139err_bus_clk:
140 clk_disable_unprepare(drv->core_clk);
141err_core_clk:
142 clk_reset(drv->core_clk, CLK_RESET_ASSERT);
143err_reset:
144 return ret;
145}
146EXPORT_SYMBOL(pil_q6v5_enable_clks);
147
148void pil_q6v5_disable_clks(struct pil_desc *pil)
149{
150 struct q6v5_data *drv = dev_get_drvdata(pil->dev);
151
152 clk_disable_unprepare(drv->bus_clk);
153 clk_disable_unprepare(drv->core_clk);
Matt Wagantall4e2599e2012-03-21 22:31:35 -0700154 clk_disable_unprepare(drv->mem_clk);
Matt Wagantalld41ce772012-05-10 23:16:41 -0700155 clk_reset(drv->core_clk, CLK_RESET_ASSERT);
156}
157EXPORT_SYMBOL(pil_q6v5_disable_clks);
158
Matt Wagantallb3fe8992011-12-07 19:26:55 -0800159void pil_q6v5_shutdown(struct pil_desc *pil)
160{
161 u32 val;
162 struct q6v5_data *drv = dev_get_drvdata(pil->dev);
163
164 /* Turn off core clock */
165 val = readl_relaxed(drv->reg_base + QDSP6SS_GFMUX_CTL);
166 val &= ~Q6SS_CLK_ENA;
167 writel_relaxed(val, drv->reg_base + QDSP6SS_GFMUX_CTL);
168
169 /* Clamp IO */
170 val = readl_relaxed(drv->reg_base + QDSP6SS_PWR_CTL);
171 val |= Q6SS_CLAMP_IO;
172 writel_relaxed(val, drv->reg_base + QDSP6SS_PWR_CTL);
173
174 /* Turn off Q6 memories */
175 val &= ~(Q6SS_L2DATA_SLP_NRET_N | Q6SS_SLP_RET_N |
176 Q6SS_L2TAG_SLP_NRET_N | Q6SS_ETB_SLP_NRET_N |
177 Q6SS_L2DATA_STBY_N);
178 writel_relaxed(Q6SS_CLAMP_IO, drv->reg_base + QDSP6SS_PWR_CTL);
179
180 /* Assert Q6 resets */
181 val = readl_relaxed(drv->reg_base + QDSP6SS_RESET);
Matt Wagantall11c07e22012-08-09 16:14:07 -0700182 val = (Q6SS_CORE_ARES | Q6SS_BUS_ARES_ENA);
Matt Wagantallb3fe8992011-12-07 19:26:55 -0800183 writel_relaxed(val, drv->reg_base + QDSP6SS_RESET);
184
185 /* Kill power at block headswitch (affects LPASS only) */
186 val = readl_relaxed(drv->reg_base + QDSP6SS_PWR_CTL);
187 val &= ~QDSS_BHS_ON;
188 writel_relaxed(val, drv->reg_base + QDSP6SS_PWR_CTL);
189}
190EXPORT_SYMBOL(pil_q6v5_shutdown);
191
192int pil_q6v5_reset(struct pil_desc *pil)
193{
194 struct q6v5_data *drv = dev_get_drvdata(pil->dev);
195 u32 val;
196
197 /* Assert resets, stop core */
198 val = readl_relaxed(drv->reg_base + QDSP6SS_RESET);
Matt Wagantall11c07e22012-08-09 16:14:07 -0700199 val |= (Q6SS_CORE_ARES | Q6SS_BUS_ARES_ENA | Q6SS_STOP_CORE);
Matt Wagantallb3fe8992011-12-07 19:26:55 -0800200 writel_relaxed(val, drv->reg_base + QDSP6SS_RESET);
201
202 /* Enable power block headswitch (only affects LPASS) */
203 val = readl_relaxed(drv->reg_base + QDSP6SS_PWR_CTL);
204 val |= QDSS_BHS_ON;
205 writel_relaxed(val, drv->reg_base + QDSP6SS_PWR_CTL);
206
207 /* Turn on memories */
208 val = readl_relaxed(drv->reg_base + QDSP6SS_PWR_CTL);
209 val |= Q6SS_L2DATA_SLP_NRET_N | Q6SS_SLP_RET_N |
210 Q6SS_L2TAG_SLP_NRET_N | Q6SS_ETB_SLP_NRET_N |
211 Q6SS_L2DATA_STBY_N;
212 writel_relaxed(val, drv->reg_base + QDSP6SS_PWR_CTL);
213
214 /* Remove IO clamp */
215 val &= ~Q6SS_CLAMP_IO;
216 writel_relaxed(val, drv->reg_base + QDSP6SS_PWR_CTL);
217
218 /* Bring core out of reset */
Matt Wagantall11c07e22012-08-09 16:14:07 -0700219 val = readl_relaxed(drv->reg_base + QDSP6SS_RESET);
220 val &= ~Q6SS_CORE_ARES;
Matt Wagantallb3fe8992011-12-07 19:26:55 -0800221 writel_relaxed(val, drv->reg_base + QDSP6SS_RESET);
222
Matt Wagantallb76f1902012-08-09 14:51:28 -0700223 /* Disable clock gating for core and rclk */
224 val = readl_relaxed(drv->reg_base + QDSP6SS_CGC_OVERRIDE);
225 val |= Q6SS_CORE_RCLK_EN | Q6SS_CORE_CLK_EN;
226 writel_relaxed(val, drv->reg_base + QDSP6SS_CGC_OVERRIDE);
227
Matt Wagantallb3fe8992011-12-07 19:26:55 -0800228 /* Turn on core clock */
229 val = readl_relaxed(drv->reg_base + QDSP6SS_GFMUX_CTL);
230 val |= Q6SS_CLK_ENA;
231 writel_relaxed(val, drv->reg_base + QDSP6SS_GFMUX_CTL);
232
233 /* Start core execution */
234 val = readl_relaxed(drv->reg_base + QDSP6SS_RESET);
235 val &= ~Q6SS_STOP_CORE;
236 writel_relaxed(val, drv->reg_base + QDSP6SS_RESET);
237
238 return 0;
239}
240EXPORT_SYMBOL(pil_q6v5_reset);
241
242struct pil_desc __devinit *pil_q6v5_init(struct platform_device *pdev)
243{
244 struct q6v5_data *drv;
245 struct resource *res;
246 struct pil_desc *desc;
247 int ret;
248
249 drv = devm_kzalloc(&pdev->dev, sizeof(*drv), GFP_KERNEL);
250 if (!drv)
251 return ERR_PTR(-ENOMEM);
252 platform_set_drvdata(pdev, drv);
253
254 res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
255 if (!res)
256 return ERR_PTR(-EINVAL);
257 drv->reg_base = devm_ioremap(&pdev->dev, res->start,
258 resource_size(res));
259 if (!drv->reg_base)
260 return ERR_PTR(-ENOMEM);
Matt Wagantallb7747992012-05-11 19:37:51 -0700261 res = platform_get_resource(pdev, IORESOURCE_MEM, 1);
262 drv->axi_halt_base = devm_ioremap(&pdev->dev, res->start,
263 resource_size(res));
264 if (!drv->axi_halt_base)
265 return ERR_PTR(-ENOMEM);
Matt Wagantallb3fe8992011-12-07 19:26:55 -0800266
Matt Wagantallb3fe8992011-12-07 19:26:55 -0800267 desc = devm_kzalloc(&pdev->dev, sizeof(*desc), GFP_KERNEL);
268 if (!desc)
269 return ERR_PTR(-ENOMEM);
270
271 ret = of_property_read_string(pdev->dev.of_node, "qcom,firmware-name",
272 &desc->name);
273 if (ret)
274 return ERR_PTR(ret);
275
276 drv->xo = devm_clk_get(&pdev->dev, "xo");
277 if (IS_ERR(drv->xo))
278 return ERR_CAST(drv->xo);
279
Matt Wagantalld41ce772012-05-10 23:16:41 -0700280 drv->bus_clk = devm_clk_get(&pdev->dev, "bus_clk");
281 if (IS_ERR(drv->bus_clk))
282 return ERR_CAST(drv->bus_clk);
283
284 drv->core_clk = devm_clk_get(&pdev->dev, "core_clk");
285 if (IS_ERR(drv->core_clk))
286 return ERR_CAST(drv->core_clk);
287
Matt Wagantallb3fe8992011-12-07 19:26:55 -0800288 desc->dev = &pdev->dev;
289
290 return desc;
291}
292EXPORT_SYMBOL(pil_q6v5_init);