blob: 40f23dd8478c48200e0c668bf47239188ad25916 [file] [log] [blame]
Dimitris Papastamos28644c802011-09-19 14:34:02 +01001/*
2 * Register cache access API - rbtree caching support
3 *
4 * Copyright 2011 Wolfson Microelectronics plc
5 *
6 * Author: Dimitris Papastamos <dp@opensource.wolfsonmicro.com>
7 *
8 * This program is free software; you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License version 2 as
10 * published by the Free Software Foundation.
11 */
12
13#include <linux/slab.h>
14#include <linux/rbtree.h>
15
16#include "internal.h"
17
18static int regcache_rbtree_write(struct regmap *map, unsigned int reg,
19 unsigned int value);
20
21struct regcache_rbtree_node {
22 /* the actual rbtree node holding this block */
23 struct rb_node node;
24 /* base register handled by this block */
25 unsigned int base_reg;
Dimitris Papastamos28644c802011-09-19 14:34:02 +010026 /* block of adjacent registers */
27 void *block;
28 /* number of registers available in the block */
29 unsigned int blklen;
30} __attribute__ ((packed));
31
32struct regcache_rbtree_ctx {
33 struct rb_root root;
34 struct regcache_rbtree_node *cached_rbnode;
35};
36
37static inline void regcache_rbtree_get_base_top_reg(
38 struct regcache_rbtree_node *rbnode,
39 unsigned int *base, unsigned int *top)
40{
41 *base = rbnode->base_reg;
42 *top = rbnode->base_reg + rbnode->blklen - 1;
43}
44
45static unsigned int regcache_rbtree_get_register(
Dimitris Papastamos25ed1152011-09-27 11:25:07 +010046 struct regcache_rbtree_node *rbnode, unsigned int idx,
47 unsigned int word_size)
Dimitris Papastamos28644c802011-09-19 14:34:02 +010048{
Lars-Peter Clausenc5713002011-09-27 20:15:37 +020049 return regcache_get_val(rbnode->block, idx, word_size);
Dimitris Papastamos28644c802011-09-19 14:34:02 +010050}
51
52static void regcache_rbtree_set_register(struct regcache_rbtree_node *rbnode,
Dimitris Papastamos25ed1152011-09-27 11:25:07 +010053 unsigned int idx, unsigned int val,
54 unsigned int word_size)
Dimitris Papastamos28644c802011-09-19 14:34:02 +010055{
Lars-Peter Clausenc5713002011-09-27 20:15:37 +020056 regcache_set_val(rbnode->block, idx, val, word_size);
Dimitris Papastamos28644c802011-09-19 14:34:02 +010057}
58
Lars-Peter Clausen3405add2011-09-27 20:15:38 +020059static struct regcache_rbtree_node *regcache_rbtree_lookup(struct regmap *map,
60 unsigned int reg)
Dimitris Papastamos28644c802011-09-19 14:34:02 +010061{
Lars-Peter Clausen3405add2011-09-27 20:15:38 +020062 struct regcache_rbtree_ctx *rbtree_ctx = map->cache;
Dimitris Papastamos28644c802011-09-19 14:34:02 +010063 struct rb_node *node;
64 struct regcache_rbtree_node *rbnode;
65 unsigned int base_reg, top_reg;
66
Lars-Peter Clausen3405add2011-09-27 20:15:38 +020067 rbnode = rbtree_ctx->cached_rbnode;
68 if (rbnode) {
Dimitris Papastamos28644c802011-09-19 14:34:02 +010069 regcache_rbtree_get_base_top_reg(rbnode, &base_reg, &top_reg);
70 if (reg >= base_reg && reg <= top_reg)
71 return rbnode;
Lars-Peter Clausen3405add2011-09-27 20:15:38 +020072 }
73
74 node = rbtree_ctx->root.rb_node;
75 while (node) {
76 rbnode = container_of(node, struct regcache_rbtree_node, node);
77 regcache_rbtree_get_base_top_reg(rbnode, &base_reg, &top_reg);
78 if (reg >= base_reg && reg <= top_reg) {
79 rbtree_ctx->cached_rbnode = rbnode;
80 return rbnode;
81 } else if (reg > top_reg) {
Dimitris Papastamos28644c802011-09-19 14:34:02 +010082 node = node->rb_right;
Lars-Peter Clausen3405add2011-09-27 20:15:38 +020083 } else if (reg < base_reg) {
Dimitris Papastamos28644c802011-09-19 14:34:02 +010084 node = node->rb_left;
Lars-Peter Clausen3405add2011-09-27 20:15:38 +020085 }
Dimitris Papastamos28644c802011-09-19 14:34:02 +010086 }
87
88 return NULL;
89}
90
91static int regcache_rbtree_insert(struct rb_root *root,
92 struct regcache_rbtree_node *rbnode)
93{
94 struct rb_node **new, *parent;
95 struct regcache_rbtree_node *rbnode_tmp;
96 unsigned int base_reg_tmp, top_reg_tmp;
97 unsigned int base_reg;
98
99 parent = NULL;
100 new = &root->rb_node;
101 while (*new) {
102 rbnode_tmp = container_of(*new, struct regcache_rbtree_node,
103 node);
104 /* base and top registers of the current rbnode */
105 regcache_rbtree_get_base_top_reg(rbnode_tmp, &base_reg_tmp,
106 &top_reg_tmp);
107 /* base register of the rbnode to be added */
108 base_reg = rbnode->base_reg;
109 parent = *new;
110 /* if this register has already been inserted, just return */
111 if (base_reg >= base_reg_tmp &&
112 base_reg <= top_reg_tmp)
113 return 0;
114 else if (base_reg > top_reg_tmp)
115 new = &((*new)->rb_right);
116 else if (base_reg < base_reg_tmp)
117 new = &((*new)->rb_left);
118 }
119
120 /* insert the node into the rbtree */
121 rb_link_node(&rbnode->node, parent, new);
122 rb_insert_color(&rbnode->node, root);
123
124 return 1;
125}
126
127static int regcache_rbtree_init(struct regmap *map)
128{
129 struct regcache_rbtree_ctx *rbtree_ctx;
130 int i;
131 int ret;
132
133 map->cache = kmalloc(sizeof *rbtree_ctx, GFP_KERNEL);
134 if (!map->cache)
135 return -ENOMEM;
136
137 rbtree_ctx = map->cache;
138 rbtree_ctx->root = RB_ROOT;
139 rbtree_ctx->cached_rbnode = NULL;
140
141 for (i = 0; i < map->num_reg_defaults; i++) {
142 ret = regcache_rbtree_write(map,
143 map->reg_defaults[i].reg,
144 map->reg_defaults[i].def);
145 if (ret)
146 goto err;
147 }
148
149 return 0;
150
151err:
152 regcache_exit(map);
153 return ret;
154}
155
156static int regcache_rbtree_exit(struct regmap *map)
157{
158 struct rb_node *next;
159 struct regcache_rbtree_ctx *rbtree_ctx;
160 struct regcache_rbtree_node *rbtree_node;
161
162 /* if we've already been called then just return */
163 rbtree_ctx = map->cache;
164 if (!rbtree_ctx)
165 return 0;
166
167 /* free up the rbtree */
168 next = rb_first(&rbtree_ctx->root);
169 while (next) {
170 rbtree_node = rb_entry(next, struct regcache_rbtree_node, node);
171 next = rb_next(&rbtree_node->node);
172 rb_erase(&rbtree_node->node, &rbtree_ctx->root);
173 kfree(rbtree_node->block);
174 kfree(rbtree_node);
175 }
176
177 /* release the resources */
178 kfree(map->cache);
179 map->cache = NULL;
180
181 return 0;
182}
183
184static int regcache_rbtree_read(struct regmap *map,
185 unsigned int reg, unsigned int *value)
186{
Dimitris Papastamos28644c802011-09-19 14:34:02 +0100187 struct regcache_rbtree_node *rbnode;
Dimitris Papastamos28644c802011-09-19 14:34:02 +0100188 unsigned int reg_tmp;
189
Lars-Peter Clausen3405add2011-09-27 20:15:38 +0200190 rbnode = regcache_rbtree_lookup(map, reg);
Dimitris Papastamos28644c802011-09-19 14:34:02 +0100191 if (rbnode) {
192 reg_tmp = reg - rbnode->base_reg;
Dimitris Papastamos25ed1152011-09-27 11:25:07 +0100193 *value = regcache_rbtree_get_register(rbnode, reg_tmp,
194 map->cache_word_size);
Dimitris Papastamos28644c802011-09-19 14:34:02 +0100195 } else {
196 /* uninitialized registers default to 0 */
197 *value = 0;
198 }
199
200 return 0;
201}
202
203
204static int regcache_rbtree_insert_to_block(struct regcache_rbtree_node *rbnode,
205 unsigned int pos, unsigned int reg,
Dimitris Papastamos25ed1152011-09-27 11:25:07 +0100206 unsigned int value, unsigned int word_size)
Dimitris Papastamos28644c802011-09-19 14:34:02 +0100207{
208 u8 *blk;
209
210 blk = krealloc(rbnode->block,
Dimitris Papastamos25ed1152011-09-27 11:25:07 +0100211 (rbnode->blklen + 1) * word_size, GFP_KERNEL);
Dimitris Papastamos28644c802011-09-19 14:34:02 +0100212 if (!blk)
213 return -ENOMEM;
214
215 /* insert the register value in the correct place in the rbnode block */
Dimitris Papastamos25ed1152011-09-27 11:25:07 +0100216 memmove(blk + (pos + 1) * word_size,
217 blk + pos * word_size,
218 (rbnode->blklen - pos) * word_size);
Dimitris Papastamos28644c802011-09-19 14:34:02 +0100219
220 /* update the rbnode block, its size and the base register */
221 rbnode->block = blk;
222 rbnode->blklen++;
223 if (!pos)
224 rbnode->base_reg = reg;
225
Dimitris Papastamos25ed1152011-09-27 11:25:07 +0100226 regcache_rbtree_set_register(rbnode, pos, value, word_size);
Dimitris Papastamos28644c802011-09-19 14:34:02 +0100227 return 0;
228}
229
230static int regcache_rbtree_write(struct regmap *map, unsigned int reg,
231 unsigned int value)
232{
233 struct regcache_rbtree_ctx *rbtree_ctx;
234 struct regcache_rbtree_node *rbnode, *rbnode_tmp;
235 struct rb_node *node;
236 unsigned int val;
237 unsigned int reg_tmp;
Dimitris Papastamos28644c802011-09-19 14:34:02 +0100238 unsigned int pos;
239 int i;
240 int ret;
241
242 rbtree_ctx = map->cache;
Dimitris Papastamos28644c802011-09-19 14:34:02 +0100243 /* if we can't locate it in the cached rbnode we'll have
244 * to traverse the rbtree looking for it.
245 */
Lars-Peter Clausen3405add2011-09-27 20:15:38 +0200246 rbnode = regcache_rbtree_lookup(map, reg);
Dimitris Papastamos28644c802011-09-19 14:34:02 +0100247 if (rbnode) {
248 reg_tmp = reg - rbnode->base_reg;
Dimitris Papastamos25ed1152011-09-27 11:25:07 +0100249 val = regcache_rbtree_get_register(rbnode, reg_tmp,
250 map->cache_word_size);
Dimitris Papastamos28644c802011-09-19 14:34:02 +0100251 if (val == value)
252 return 0;
Dimitris Papastamos25ed1152011-09-27 11:25:07 +0100253 regcache_rbtree_set_register(rbnode, reg_tmp, value,
254 map->cache_word_size);
Dimitris Papastamos28644c802011-09-19 14:34:02 +0100255 } else {
256 /* bail out early, no need to create the rbnode yet */
257 if (!value)
258 return 0;
259 /* look for an adjacent register to the one we are about to add */
260 for (node = rb_first(&rbtree_ctx->root); node;
261 node = rb_next(node)) {
262 rbnode_tmp = rb_entry(node, struct regcache_rbtree_node, node);
263 for (i = 0; i < rbnode_tmp->blklen; i++) {
264 reg_tmp = rbnode_tmp->base_reg + i;
265 if (abs(reg_tmp - reg) != 1)
266 continue;
267 /* decide where in the block to place our register */
268 if (reg_tmp + 1 == reg)
269 pos = i + 1;
270 else
271 pos = i;
272 ret = regcache_rbtree_insert_to_block(rbnode_tmp, pos,
Dimitris Papastamos25ed1152011-09-27 11:25:07 +0100273 reg, value,
274 map->cache_word_size);
Dimitris Papastamos28644c802011-09-19 14:34:02 +0100275 if (ret)
276 return ret;
277 rbtree_ctx->cached_rbnode = rbnode_tmp;
278 return 0;
279 }
280 }
281 /* we did not manage to find a place to insert it in an existing
282 * block so create a new rbnode with a single register in its block.
283 * This block will get populated further if any other adjacent
284 * registers get modified in the future.
285 */
286 rbnode = kzalloc(sizeof *rbnode, GFP_KERNEL);
287 if (!rbnode)
288 return -ENOMEM;
289 rbnode->blklen = 1;
290 rbnode->base_reg = reg;
Dimitris Papastamos25ed1152011-09-27 11:25:07 +0100291 rbnode->block = kmalloc(rbnode->blklen * map->cache_word_size,
Dimitris Papastamos28644c802011-09-19 14:34:02 +0100292 GFP_KERNEL);
293 if (!rbnode->block) {
294 kfree(rbnode);
295 return -ENOMEM;
296 }
Dimitris Papastamos25ed1152011-09-27 11:25:07 +0100297 regcache_rbtree_set_register(rbnode, 0, value, map->cache_word_size);
Dimitris Papastamos28644c802011-09-19 14:34:02 +0100298 regcache_rbtree_insert(&rbtree_ctx->root, rbnode);
299 rbtree_ctx->cached_rbnode = rbnode;
300 }
301
302 return 0;
303}
304
305static int regcache_rbtree_sync(struct regmap *map)
306{
307 struct regcache_rbtree_ctx *rbtree_ctx;
308 struct rb_node *node;
309 struct regcache_rbtree_node *rbnode;
310 unsigned int regtmp;
311 unsigned int val, def;
312 int ret;
313 int i;
314
315 rbtree_ctx = map->cache;
316 for (node = rb_first(&rbtree_ctx->root); node; node = rb_next(node)) {
317 rbnode = rb_entry(node, struct regcache_rbtree_node, node);
318 for (i = 0; i < rbnode->blklen; i++) {
319 regtmp = rbnode->base_reg + i;
Dimitris Papastamos25ed1152011-09-27 11:25:07 +0100320 val = regcache_rbtree_get_register(rbnode, i,
321 map->cache_word_size);
Dimitris Papastamos28644c802011-09-19 14:34:02 +0100322 ret = regcache_lookup_reg(map, i);
323 if (ret < 0)
324 def = 0;
325 else
326 def = map->reg_defaults[ret].def;
327 if (val == def)
328 continue;
329 map->cache_bypass = 1;
Dimitris Papastamos13753a92011-09-29 14:36:25 +0100330 ret = _regmap_write(map, regtmp, val);
Dimitris Papastamos28644c802011-09-19 14:34:02 +0100331 map->cache_bypass = 0;
332 if (ret)
333 return ret;
334 dev_dbg(map->dev, "Synced register %#x, value %#x\n",
335 regtmp, val);
336 }
337 }
338
339 return 0;
340}
341
342struct regcache_ops regcache_rbtree_ops = {
343 .type = REGCACHE_RBTREE,
344 .name = "rbtree",
345 .init = regcache_rbtree_init,
346 .exit = regcache_rbtree_exit,
347 .read = regcache_rbtree_read,
348 .write = regcache_rbtree_write,
349 .sync = regcache_rbtree_sync
350};