blob: 3b9d5420b774c3f42ae3628a906607c1585ebed7 [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>
18#include <linux/elf.h>
19#include <linux/err.h>
20#include <linux/of.h>
21#include <linux/clk.h>
22
Matt Wagantalld41ce772012-05-10 23:16:41 -070023#include <mach/clk.h>
24
Matt Wagantallb3fe8992011-12-07 19:26:55 -080025#include "peripheral-loader.h"
26#include "pil-q6v5.h"
27
28/* Register Offsets */
29#define QDSP6SS_RESET 0x014
30#define QDSP6SS_GFMUX_CTL 0x020
31#define QDSP6SS_PWR_CTL 0x030
32
33/* QDSP6SS_RESET */
34#define Q6SS_CORE_ARES BIT(1)
35#define Q6SS_ETM_ISDB_ARES BIT(3)
36#define Q6SS_STOP_CORE BIT(4)
37
38/* QDSP6SS_GFMUX_CTL */
39#define Q6SS_CLK_ENA BIT(1)
40
41/* QDSP6SS_PWR_CTL */
42#define Q6SS_L2DATA_SLP_NRET_N BIT(0)
43#define Q6SS_L2TAG_SLP_NRET_N BIT(16)
44#define Q6SS_ETB_SLP_NRET_N BIT(17)
45#define Q6SS_L2DATA_STBY_N BIT(18)
46#define Q6SS_SLP_RET_N BIT(19)
47#define Q6SS_CLAMP_IO BIT(20)
48#define QDSS_BHS_ON BIT(21)
49
50int pil_q6v5_make_proxy_votes(struct pil_desc *pil)
51{
52 int ret;
53 struct q6v5_data *drv = dev_get_drvdata(pil->dev);
54
55 ret = clk_prepare_enable(drv->xo);
56 if (ret) {
57 dev_err(pil->dev, "Failed to enable XO\n");
58 return ret;
59 }
60 return 0;
61}
62EXPORT_SYMBOL(pil_q6v5_make_proxy_votes);
63
64void pil_q6v5_remove_proxy_votes(struct pil_desc *pil)
65{
66 struct q6v5_data *drv = dev_get_drvdata(pil->dev);
67 clk_disable_unprepare(drv->xo);
68}
69EXPORT_SYMBOL(pil_q6v5_remove_proxy_votes);
70
71int pil_q6v5_init_image(struct pil_desc *pil, const u8 *metadata,
72 size_t size)
73{
74 const struct elf32_hdr *ehdr = (struct elf32_hdr *)metadata;
75 struct q6v5_data *drv = dev_get_drvdata(pil->dev);
76 drv->start_addr = ehdr->e_entry;
77 return 0;
78}
79EXPORT_SYMBOL(pil_q6v5_init_image);
80
Matt Wagantalld41ce772012-05-10 23:16:41 -070081int pil_q6v5_enable_clks(struct pil_desc *pil)
82{
83 struct q6v5_data *drv = dev_get_drvdata(pil->dev);
84 int ret;
85
86 ret = clk_reset(drv->core_clk, CLK_RESET_DEASSERT);
87 if (ret)
88 goto err_reset;
89 ret = clk_prepare_enable(drv->core_clk);
90 if (ret)
91 goto err_core_clk;
92 ret = clk_prepare_enable(drv->bus_clk);
93 if (ret)
94 goto err_bus_clk;
95
96 return 0;
97
98err_bus_clk:
99 clk_disable_unprepare(drv->core_clk);
100err_core_clk:
101 clk_reset(drv->core_clk, CLK_RESET_ASSERT);
102err_reset:
103 return ret;
104}
105EXPORT_SYMBOL(pil_q6v5_enable_clks);
106
107void pil_q6v5_disable_clks(struct pil_desc *pil)
108{
109 struct q6v5_data *drv = dev_get_drvdata(pil->dev);
110
111 clk_disable_unprepare(drv->bus_clk);
112 clk_disable_unprepare(drv->core_clk);
113 clk_reset(drv->core_clk, CLK_RESET_ASSERT);
114}
115EXPORT_SYMBOL(pil_q6v5_disable_clks);
116
Matt Wagantallb3fe8992011-12-07 19:26:55 -0800117void pil_q6v5_shutdown(struct pil_desc *pil)
118{
119 u32 val;
120 struct q6v5_data *drv = dev_get_drvdata(pil->dev);
121
122 /* Turn off core clock */
123 val = readl_relaxed(drv->reg_base + QDSP6SS_GFMUX_CTL);
124 val &= ~Q6SS_CLK_ENA;
125 writel_relaxed(val, drv->reg_base + QDSP6SS_GFMUX_CTL);
126
127 /* Clamp IO */
128 val = readl_relaxed(drv->reg_base + QDSP6SS_PWR_CTL);
129 val |= Q6SS_CLAMP_IO;
130 writel_relaxed(val, drv->reg_base + QDSP6SS_PWR_CTL);
131
132 /* Turn off Q6 memories */
133 val &= ~(Q6SS_L2DATA_SLP_NRET_N | Q6SS_SLP_RET_N |
134 Q6SS_L2TAG_SLP_NRET_N | Q6SS_ETB_SLP_NRET_N |
135 Q6SS_L2DATA_STBY_N);
136 writel_relaxed(Q6SS_CLAMP_IO, drv->reg_base + QDSP6SS_PWR_CTL);
137
138 /* Assert Q6 resets */
139 val = readl_relaxed(drv->reg_base + QDSP6SS_RESET);
140 val = (Q6SS_CORE_ARES | Q6SS_ETM_ISDB_ARES);
141 writel_relaxed(val, drv->reg_base + QDSP6SS_RESET);
142
143 /* Kill power at block headswitch (affects LPASS only) */
144 val = readl_relaxed(drv->reg_base + QDSP6SS_PWR_CTL);
145 val &= ~QDSS_BHS_ON;
146 writel_relaxed(val, drv->reg_base + QDSP6SS_PWR_CTL);
147}
148EXPORT_SYMBOL(pil_q6v5_shutdown);
149
150int pil_q6v5_reset(struct pil_desc *pil)
151{
152 struct q6v5_data *drv = dev_get_drvdata(pil->dev);
153 u32 val;
154
155 /* Assert resets, stop core */
156 val = readl_relaxed(drv->reg_base + QDSP6SS_RESET);
157 val |= (Q6SS_CORE_ARES | Q6SS_ETM_ISDB_ARES | Q6SS_STOP_CORE);
158 writel_relaxed(val, drv->reg_base + QDSP6SS_RESET);
159
160 /* Enable power block headswitch (only affects LPASS) */
161 val = readl_relaxed(drv->reg_base + QDSP6SS_PWR_CTL);
162 val |= QDSS_BHS_ON;
163 writel_relaxed(val, drv->reg_base + QDSP6SS_PWR_CTL);
164
165 /* Turn on memories */
166 val = readl_relaxed(drv->reg_base + QDSP6SS_PWR_CTL);
167 val |= Q6SS_L2DATA_SLP_NRET_N | Q6SS_SLP_RET_N |
168 Q6SS_L2TAG_SLP_NRET_N | Q6SS_ETB_SLP_NRET_N |
169 Q6SS_L2DATA_STBY_N;
170 writel_relaxed(val, drv->reg_base + QDSP6SS_PWR_CTL);
171
172 /* Remove IO clamp */
173 val &= ~Q6SS_CLAMP_IO;
174 writel_relaxed(val, drv->reg_base + QDSP6SS_PWR_CTL);
175
176 /* Bring core out of reset */
177 val = Q6SS_STOP_CORE;
178 writel_relaxed(val, drv->reg_base + QDSP6SS_RESET);
179
180 /* Turn on core clock */
181 val = readl_relaxed(drv->reg_base + QDSP6SS_GFMUX_CTL);
182 val |= Q6SS_CLK_ENA;
183 writel_relaxed(val, drv->reg_base + QDSP6SS_GFMUX_CTL);
184
185 /* Start core execution */
186 val = readl_relaxed(drv->reg_base + QDSP6SS_RESET);
187 val &= ~Q6SS_STOP_CORE;
188 writel_relaxed(val, drv->reg_base + QDSP6SS_RESET);
189
190 return 0;
191}
192EXPORT_SYMBOL(pil_q6v5_reset);
193
194struct pil_desc __devinit *pil_q6v5_init(struct platform_device *pdev)
195{
196 struct q6v5_data *drv;
197 struct resource *res;
198 struct pil_desc *desc;
199 int ret;
200
201 drv = devm_kzalloc(&pdev->dev, sizeof(*drv), GFP_KERNEL);
202 if (!drv)
203 return ERR_PTR(-ENOMEM);
204 platform_set_drvdata(pdev, drv);
205
206 res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
207 if (!res)
208 return ERR_PTR(-EINVAL);
209 drv->reg_base = devm_ioremap(&pdev->dev, res->start,
210 resource_size(res));
211 if (!drv->reg_base)
212 return ERR_PTR(-ENOMEM);
213
Matt Wagantallb3fe8992011-12-07 19:26:55 -0800214 desc = devm_kzalloc(&pdev->dev, sizeof(*desc), GFP_KERNEL);
215 if (!desc)
216 return ERR_PTR(-ENOMEM);
217
218 ret = of_property_read_string(pdev->dev.of_node, "qcom,firmware-name",
219 &desc->name);
220 if (ret)
221 return ERR_PTR(ret);
222
223 drv->xo = devm_clk_get(&pdev->dev, "xo");
224 if (IS_ERR(drv->xo))
225 return ERR_CAST(drv->xo);
226
Matt Wagantalld41ce772012-05-10 23:16:41 -0700227 drv->bus_clk = devm_clk_get(&pdev->dev, "bus_clk");
228 if (IS_ERR(drv->bus_clk))
229 return ERR_CAST(drv->bus_clk);
230
231 drv->core_clk = devm_clk_get(&pdev->dev, "core_clk");
232 if (IS_ERR(drv->core_clk))
233 return ERR_CAST(drv->core_clk);
234
Matt Wagantallb3fe8992011-12-07 19:26:55 -0800235 desc->dev = &pdev->dev;
236
237 return desc;
238}
239EXPORT_SYMBOL(pil_q6v5_init);