blob: 1af2b0e0182ffb338466e3a0abbb1990e66bdcdf [file] [log] [blame]
Pawel Moll3ecbf052012-09-24 14:55:40 +01001/*
2 * This program is free software; you can redistribute it and/or modify
3 * it under the terms of the GNU General Public License version 2 as
4 * published by the Free Software Foundation.
5 *
6 * This program is distributed in the hope that it will be useful,
7 * but WITHOUT ANY WARRANTY; without even the implied warranty of
8 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
9 * GNU General Public License for more details.
10 *
11 * Copyright (C) 2012 ARM Limited
12 */
13
14#define pr_fmt(fmt) "vexpress-config: " fmt
15
16#include <linux/bitops.h>
17#include <linux/completion.h>
18#include <linux/export.h>
19#include <linux/init.h>
20#include <linux/list.h>
21#include <linux/of.h>
22#include <linux/of_device.h>
23#include <linux/slab.h>
24#include <linux/string.h>
25#include <linux/vexpress.h>
26
27
28#define VEXPRESS_CONFIG_MAX_BRIDGES 2
29
30struct vexpress_config_bridge {
31 struct device_node *node;
32 struct vexpress_config_bridge_info *info;
33 struct list_head transactions;
34 spinlock_t transactions_lock;
35} vexpress_config_bridges[VEXPRESS_CONFIG_MAX_BRIDGES];
36
37static DECLARE_BITMAP(vexpress_config_bridges_map,
38 ARRAY_SIZE(vexpress_config_bridges));
39static DEFINE_MUTEX(vexpress_config_bridges_mutex);
40
41struct vexpress_config_bridge *vexpress_config_bridge_register(
42 struct device_node *node,
43 struct vexpress_config_bridge_info *info)
44{
45 struct vexpress_config_bridge *bridge;
46 int i;
47
48 pr_debug("Registering bridge '%s'\n", info->name);
49
50 mutex_lock(&vexpress_config_bridges_mutex);
51 i = find_first_zero_bit(vexpress_config_bridges_map,
52 ARRAY_SIZE(vexpress_config_bridges));
53 if (i >= ARRAY_SIZE(vexpress_config_bridges)) {
54 pr_err("Can't register more bridges!\n");
55 mutex_unlock(&vexpress_config_bridges_mutex);
56 return NULL;
57 }
58 __set_bit(i, vexpress_config_bridges_map);
59 bridge = &vexpress_config_bridges[i];
60
61 bridge->node = node;
62 bridge->info = info;
63 INIT_LIST_HEAD(&bridge->transactions);
64 spin_lock_init(&bridge->transactions_lock);
65
66 mutex_unlock(&vexpress_config_bridges_mutex);
67
68 return bridge;
69}
Guenter Roecka17155b2013-01-06 01:16:19 -080070EXPORT_SYMBOL(vexpress_config_bridge_register);
Pawel Moll3ecbf052012-09-24 14:55:40 +010071
72void vexpress_config_bridge_unregister(struct vexpress_config_bridge *bridge)
73{
74 struct vexpress_config_bridge __bridge = *bridge;
75 int i;
76
77 mutex_lock(&vexpress_config_bridges_mutex);
78 for (i = 0; i < ARRAY_SIZE(vexpress_config_bridges); i++)
79 if (&vexpress_config_bridges[i] == bridge)
80 __clear_bit(i, vexpress_config_bridges_map);
81 mutex_unlock(&vexpress_config_bridges_mutex);
82
83 WARN_ON(!list_empty(&__bridge.transactions));
84 while (!list_empty(&__bridge.transactions))
85 cpu_relax();
86}
Guenter Roecka17155b2013-01-06 01:16:19 -080087EXPORT_SYMBOL(vexpress_config_bridge_unregister);
Pawel Moll3ecbf052012-09-24 14:55:40 +010088
Pawel Moll854f6232013-06-06 10:59:22 +010089static struct vexpress_config_bridge *
90 vexpress_config_bridge_find(struct device_node *node)
Pawel Moll3ecbf052012-09-24 14:55:40 +010091{
Pawel Moll3ecbf052012-09-24 14:55:40 +010092 int i;
Pawel Moll854f6232013-06-06 10:59:22 +010093 struct vexpress_config_bridge *res = NULL;
94 struct device_node *bridge_node = of_node_get(node);
Pawel Moll3ecbf052012-09-24 14:55:40 +010095
Pawel Moll3ecbf052012-09-24 14:55:40 +010096 while (bridge_node) {
97 const __be32 *prop = of_get_property(bridge_node,
98 "arm,vexpress,config-bridge", NULL);
99
100 if (prop) {
101 bridge_node = of_find_node_by_phandle(
102 be32_to_cpup(prop));
103 break;
104 }
105
106 bridge_node = of_get_next_parent(bridge_node);
107 }
108
109 mutex_lock(&vexpress_config_bridges_mutex);
110 for (i = 0; i < ARRAY_SIZE(vexpress_config_bridges); i++) {
111 struct vexpress_config_bridge *bridge =
112 &vexpress_config_bridges[i];
113
114 if (test_bit(i, vexpress_config_bridges_map) &&
115 bridge->node == bridge_node) {
Pawel Moll854f6232013-06-06 10:59:22 +0100116 res = bridge;
Pawel Moll3ecbf052012-09-24 14:55:40 +0100117 break;
118 }
119 }
120 mutex_unlock(&vexpress_config_bridges_mutex);
121
Pawel Moll854f6232013-06-06 10:59:22 +0100122 return res;
123}
124
125
126struct vexpress_config_func {
127 struct vexpress_config_bridge *bridge;
128 void *func;
129};
130
131struct vexpress_config_func *__vexpress_config_func_get(
132 struct vexpress_config_bridge *bridge,
133 struct device *dev,
134 struct device_node *node,
135 const char *id)
136{
137 struct vexpress_config_func *func;
138
139 if (WARN_ON(dev && node && dev->of_node != node))
140 return NULL;
141 if (dev && !node)
142 node = dev->of_node;
143
144 if (!bridge)
145 bridge = vexpress_config_bridge_find(node);
146 if (!bridge)
147 return NULL;
148
149 func = kzalloc(sizeof(*func), GFP_KERNEL);
150 if (!func)
151 return NULL;
152
153 func->bridge = bridge;
154 func->func = bridge->info->func_get(dev, node, id);
155
Pawel Moll3ecbf052012-09-24 14:55:40 +0100156 if (!func->func) {
157 of_node_put(node);
158 kfree(func);
159 return NULL;
160 }
161
162 return func;
163}
Guenter Roecka17155b2013-01-06 01:16:19 -0800164EXPORT_SYMBOL(__vexpress_config_func_get);
Pawel Moll3ecbf052012-09-24 14:55:40 +0100165
166void vexpress_config_func_put(struct vexpress_config_func *func)
167{
168 func->bridge->info->func_put(func->func);
169 of_node_put(func->bridge->node);
170 kfree(func);
171}
Guenter Roecka17155b2013-01-06 01:16:19 -0800172EXPORT_SYMBOL(vexpress_config_func_put);
Pawel Moll3ecbf052012-09-24 14:55:40 +0100173
174struct vexpress_config_trans {
175 struct vexpress_config_func *func;
176 int offset;
177 bool write;
178 u32 *data;
179 int status;
180 struct completion completion;
181 struct list_head list;
182};
183
184static void vexpress_config_dump_trans(const char *what,
185 struct vexpress_config_trans *trans)
186{
187 pr_debug("%s %s trans %p func 0x%p offset %d data 0x%x status %d\n",
188 what, trans->write ? "write" : "read", trans,
189 trans->func->func, trans->offset,
190 trans->data ? *trans->data : 0, trans->status);
191}
192
193static int vexpress_config_schedule(struct vexpress_config_trans *trans)
194{
195 int status;
196 struct vexpress_config_bridge *bridge = trans->func->bridge;
197 unsigned long flags;
198
199 init_completion(&trans->completion);
200 trans->status = -EFAULT;
201
202 spin_lock_irqsave(&bridge->transactions_lock, flags);
203
Pawel Moll367764a2013-04-25 18:14:51 +0100204 if (list_empty(&bridge->transactions)) {
205 vexpress_config_dump_trans("Executing", trans);
Pawel Moll3ecbf052012-09-24 14:55:40 +0100206 status = bridge->info->func_exec(trans->func->func,
207 trans->offset, trans->write, trans->data);
Pawel Moll367764a2013-04-25 18:14:51 +0100208 } else {
209 vexpress_config_dump_trans("Queuing", trans);
Pawel Moll3ecbf052012-09-24 14:55:40 +0100210 status = VEXPRESS_CONFIG_STATUS_WAIT;
Pawel Moll367764a2013-04-25 18:14:51 +0100211 }
Pawel Moll3ecbf052012-09-24 14:55:40 +0100212
213 switch (status) {
214 case VEXPRESS_CONFIG_STATUS_DONE:
215 vexpress_config_dump_trans("Finished", trans);
216 trans->status = status;
217 break;
218 case VEXPRESS_CONFIG_STATUS_WAIT:
219 list_add_tail(&trans->list, &bridge->transactions);
220 break;
221 }
222
223 spin_unlock_irqrestore(&bridge->transactions_lock, flags);
224
225 return status;
226}
227
228void vexpress_config_complete(struct vexpress_config_bridge *bridge,
229 int status)
230{
231 struct vexpress_config_trans *trans;
232 unsigned long flags;
Pawel Moll367764a2013-04-25 18:14:51 +0100233 const char *message = "Completed";
Pawel Moll3ecbf052012-09-24 14:55:40 +0100234
235 spin_lock_irqsave(&bridge->transactions_lock, flags);
236
237 trans = list_first_entry(&bridge->transactions,
238 struct vexpress_config_trans, list);
Pawel Moll3ecbf052012-09-24 14:55:40 +0100239 trans->status = status;
Pawel Moll3ecbf052012-09-24 14:55:40 +0100240
Pawel Moll367764a2013-04-25 18:14:51 +0100241 do {
242 vexpress_config_dump_trans(message, trans);
243 list_del(&trans->list);
244 complete(&trans->completion);
Pawel Moll3ecbf052012-09-24 14:55:40 +0100245
Pawel Moll367764a2013-04-25 18:14:51 +0100246 if (list_empty(&bridge->transactions))
247 break;
248
249 trans = list_first_entry(&bridge->transactions,
250 struct vexpress_config_trans, list);
251 vexpress_config_dump_trans("Executing pending", trans);
252 trans->status = bridge->info->func_exec(trans->func->func,
253 trans->offset, trans->write, trans->data);
254 message = "Finished pending";
255 } while (trans->status == VEXPRESS_CONFIG_STATUS_DONE);
256
Pawel Moll3ecbf052012-09-24 14:55:40 +0100257 spin_unlock_irqrestore(&bridge->transactions_lock, flags);
Pawel Moll3ecbf052012-09-24 14:55:40 +0100258}
Guenter Roecka17155b2013-01-06 01:16:19 -0800259EXPORT_SYMBOL(vexpress_config_complete);
Pawel Moll3ecbf052012-09-24 14:55:40 +0100260
261int vexpress_config_wait(struct vexpress_config_trans *trans)
262{
263 wait_for_completion(&trans->completion);
264
265 return trans->status;
266}
Guenter Roecka17155b2013-01-06 01:16:19 -0800267EXPORT_SYMBOL(vexpress_config_wait);
Pawel Moll3ecbf052012-09-24 14:55:40 +0100268
269int vexpress_config_read(struct vexpress_config_func *func, int offset,
270 u32 *data)
271{
272 struct vexpress_config_trans trans = {
273 .func = func,
274 .offset = offset,
275 .write = false,
276 .data = data,
277 .status = 0,
278 };
279 int status = vexpress_config_schedule(&trans);
280
281 if (status == VEXPRESS_CONFIG_STATUS_WAIT)
282 status = vexpress_config_wait(&trans);
283
284 return status;
285}
286EXPORT_SYMBOL(vexpress_config_read);
287
288int vexpress_config_write(struct vexpress_config_func *func, int offset,
289 u32 data)
290{
291 struct vexpress_config_trans trans = {
292 .func = func,
293 .offset = offset,
294 .write = true,
295 .data = &data,
296 .status = 0,
297 };
298 int status = vexpress_config_schedule(&trans);
299
300 if (status == VEXPRESS_CONFIG_STATUS_WAIT)
301 status = vexpress_config_wait(&trans);
302
303 return status;
304}
305EXPORT_SYMBOL(vexpress_config_write);