summaryrefslogtreecommitdiff
path: root/ANDROID_3.4.5/drivers/base/regmap
diff options
context:
space:
mode:
Diffstat (limited to 'ANDROID_3.4.5/drivers/base/regmap')
-rw-r--r--ANDROID_3.4.5/drivers/base/regmap/Kconfig18
-rw-r--r--ANDROID_3.4.5/drivers/base/regmap/Makefile6
-rw-r--r--ANDROID_3.4.5/drivers/base/regmap/internal.h130
-rw-r--r--ANDROID_3.4.5/drivers/base/regmap/regcache-lzo.c379
-rw-r--r--ANDROID_3.4.5/drivers/base/regmap/regcache-rbtree.c430
-rw-r--r--ANDROID_3.4.5/drivers/base/regmap/regcache.c491
-rw-r--r--ANDROID_3.4.5/drivers/base/regmap/regmap-debugfs.c286
-rw-r--r--ANDROID_3.4.5/drivers/base/regmap/regmap-i2c.c131
-rw-r--r--ANDROID_3.4.5/drivers/base/regmap/regmap-irq.c303
-rw-r--r--ANDROID_3.4.5/drivers/base/regmap/regmap-spi.c90
-rw-r--r--ANDROID_3.4.5/drivers/base/regmap/regmap.c936
11 files changed, 0 insertions, 3200 deletions
diff --git a/ANDROID_3.4.5/drivers/base/regmap/Kconfig b/ANDROID_3.4.5/drivers/base/regmap/Kconfig
deleted file mode 100644
index 0f6c7fb4..00000000
--- a/ANDROID_3.4.5/drivers/base/regmap/Kconfig
+++ /dev/null
@@ -1,18 +0,0 @@
-# Generic register map support. There are no user servicable options here,
-# this is an API intended to be used by other kernel subsystems. These
-# subsystems should select the appropriate symbols.
-
-config REGMAP
- default y if (REGMAP_I2C || REGMAP_SPI)
- select LZO_COMPRESS
- select LZO_DECOMPRESS
- bool
-
-config REGMAP_I2C
- tristate
-
-config REGMAP_SPI
- tristate
-
-config REGMAP_IRQ
- bool
diff --git a/ANDROID_3.4.5/drivers/base/regmap/Makefile b/ANDROID_3.4.5/drivers/base/regmap/Makefile
deleted file mode 100644
index defd5796..00000000
--- a/ANDROID_3.4.5/drivers/base/regmap/Makefile
+++ /dev/null
@@ -1,6 +0,0 @@
-obj-$(CONFIG_REGMAP) += regmap.o regcache.o
-obj-$(CONFIG_REGMAP) += regcache-rbtree.o regcache-lzo.o
-obj-$(CONFIG_DEBUG_FS) += regmap-debugfs.o
-obj-$(CONFIG_REGMAP_I2C) += regmap-i2c.o
-obj-$(CONFIG_REGMAP_SPI) += regmap-spi.o
-obj-$(CONFIG_REGMAP_IRQ) += regmap-irq.o
diff --git a/ANDROID_3.4.5/drivers/base/regmap/internal.h b/ANDROID_3.4.5/drivers/base/regmap/internal.h
deleted file mode 100644
index fcafc5b2..00000000
--- a/ANDROID_3.4.5/drivers/base/regmap/internal.h
+++ /dev/null
@@ -1,130 +0,0 @@
-/*
- * Register map access API internal header
- *
- * Copyright 2011 Wolfson Microelectronics plc
- *
- * Author: Mark Brown <broonie@opensource.wolfsonmicro.com>
- *
- * This program is free software; you can redistribute it and/or modify
- * it under the terms of the GNU General Public License version 2 as
- * published by the Free Software Foundation.
- */
-
-#ifndef _REGMAP_INTERNAL_H
-#define _REGMAP_INTERNAL_H
-
-#include <linux/regmap.h>
-#include <linux/fs.h>
-
-struct regmap;
-struct regcache_ops;
-
-struct regmap_format {
- size_t buf_size;
- size_t reg_bytes;
- size_t pad_bytes;
- size_t val_bytes;
- void (*format_write)(struct regmap *map,
- unsigned int reg, unsigned int val);
- void (*format_reg)(void *buf, unsigned int reg);
- void (*format_val)(void *buf, unsigned int val);
- unsigned int (*parse_val)(void *buf);
-};
-
-struct regmap {
- struct mutex lock;
-
- struct device *dev; /* Device we do I/O on */
- void *work_buf; /* Scratch buffer used to format I/O */
- struct regmap_format format; /* Buffer format */
- const struct regmap_bus *bus;
-
-#ifdef CONFIG_DEBUG_FS
- struct dentry *debugfs;
-#endif
-
- unsigned int max_register;
- bool (*writeable_reg)(struct device *dev, unsigned int reg);
- bool (*readable_reg)(struct device *dev, unsigned int reg);
- bool (*volatile_reg)(struct device *dev, unsigned int reg);
- bool (*precious_reg)(struct device *dev, unsigned int reg);
-
- u8 read_flag_mask;
- u8 write_flag_mask;
-
- /* regcache specific members */
- const struct regcache_ops *cache_ops;
- enum regcache_type cache_type;
-
- /* number of bytes in reg_defaults_raw */
- unsigned int cache_size_raw;
- /* number of bytes per word in reg_defaults_raw */
- unsigned int cache_word_size;
- /* number of entries in reg_defaults */
- unsigned int num_reg_defaults;
- /* number of entries in reg_defaults_raw */
- unsigned int num_reg_defaults_raw;
-
- /* if set, only the cache is modified not the HW */
- u32 cache_only;
- /* if set, only the HW is modified not the cache */
- u32 cache_bypass;
- /* if set, remember to free reg_defaults_raw */
- bool cache_free;
-
- struct reg_default *reg_defaults;
- const void *reg_defaults_raw;
- void *cache;
- u32 cache_dirty;
-
- struct reg_default *patch;
- int patch_regs;
-};
-
-struct regcache_ops {
- const char *name;
- enum regcache_type type;
- int (*init)(struct regmap *map);
- int (*exit)(struct regmap *map);
- int (*read)(struct regmap *map, unsigned int reg, unsigned int *value);
- int (*write)(struct regmap *map, unsigned int reg, unsigned int value);
- int (*sync)(struct regmap *map, unsigned int min, unsigned int max);
-};
-
-bool regmap_writeable(struct regmap *map, unsigned int reg);
-bool regmap_readable(struct regmap *map, unsigned int reg);
-bool regmap_volatile(struct regmap *map, unsigned int reg);
-bool regmap_precious(struct regmap *map, unsigned int reg);
-
-int _regmap_write(struct regmap *map, unsigned int reg,
- unsigned int val);
-
-#ifdef CONFIG_DEBUG_FS
-extern void regmap_debugfs_initcall(void);
-extern void regmap_debugfs_init(struct regmap *map);
-extern void regmap_debugfs_exit(struct regmap *map);
-#else
-static inline void regmap_debugfs_initcall(void) { }
-static inline void regmap_debugfs_init(struct regmap *map) { }
-static inline void regmap_debugfs_exit(struct regmap *map) { }
-#endif
-
-/* regcache core declarations */
-int regcache_init(struct regmap *map, const struct regmap_config *config);
-void regcache_exit(struct regmap *map);
-int regcache_read(struct regmap *map,
- unsigned int reg, unsigned int *value);
-int regcache_write(struct regmap *map,
- unsigned int reg, unsigned int value);
-int regcache_sync(struct regmap *map);
-
-unsigned int regcache_get_val(const void *base, unsigned int idx,
- unsigned int word_size);
-bool regcache_set_val(void *base, unsigned int idx,
- unsigned int val, unsigned int word_size);
-int regcache_lookup_reg(struct regmap *map, unsigned int reg);
-
-extern struct regcache_ops regcache_rbtree_ops;
-extern struct regcache_ops regcache_lzo_ops;
-
-#endif
diff --git a/ANDROID_3.4.5/drivers/base/regmap/regcache-lzo.c b/ANDROID_3.4.5/drivers/base/regmap/regcache-lzo.c
deleted file mode 100644
index 483b06d4..00000000
--- a/ANDROID_3.4.5/drivers/base/regmap/regcache-lzo.c
+++ /dev/null
@@ -1,379 +0,0 @@
-/*
- * Register cache access API - LZO caching support
- *
- * Copyright 2011 Wolfson Microelectronics plc
- *
- * Author: Dimitris Papastamos <dp@opensource.wolfsonmicro.com>
- *
- * This program is free software; you can redistribute it and/or modify
- * it under the terms of the GNU General Public License version 2 as
- * published by the Free Software Foundation.
- */
-
-#include <linux/slab.h>
-#include <linux/device.h>
-#include <linux/lzo.h>
-
-#include "internal.h"
-
-static int regcache_lzo_exit(struct regmap *map);
-
-struct regcache_lzo_ctx {
- void *wmem;
- void *dst;
- const void *src;
- size_t src_len;
- size_t dst_len;
- size_t decompressed_size;
- unsigned long *sync_bmp;
- int sync_bmp_nbits;
-};
-
-#define LZO_BLOCK_NUM 8
-static int regcache_lzo_block_count(struct regmap *map)
-{
- return LZO_BLOCK_NUM;
-}
-
-static int regcache_lzo_prepare(struct regcache_lzo_ctx *lzo_ctx)
-{
- lzo_ctx->wmem = kmalloc(LZO1X_MEM_COMPRESS, GFP_KERNEL);
- if (!lzo_ctx->wmem)
- return -ENOMEM;
- return 0;
-}
-
-static int regcache_lzo_compress(struct regcache_lzo_ctx *lzo_ctx)
-{
- size_t compress_size;
- int ret;
-
- ret = lzo1x_1_compress(lzo_ctx->src, lzo_ctx->src_len,
- lzo_ctx->dst, &compress_size, lzo_ctx->wmem);
- if (ret != LZO_E_OK || compress_size > lzo_ctx->dst_len)
- return -EINVAL;
- lzo_ctx->dst_len = compress_size;
- return 0;
-}
-
-static int regcache_lzo_decompress(struct regcache_lzo_ctx *lzo_ctx)
-{
- size_t dst_len;
- int ret;
-
- dst_len = lzo_ctx->dst_len;
- ret = lzo1x_decompress_safe(lzo_ctx->src, lzo_ctx->src_len,
- lzo_ctx->dst, &dst_len);
- if (ret != LZO_E_OK || dst_len != lzo_ctx->dst_len)
- return -EINVAL;
- return 0;
-}
-
-static int regcache_lzo_compress_cache_block(struct regmap *map,
- struct regcache_lzo_ctx *lzo_ctx)
-{
- int ret;
-
- lzo_ctx->dst_len = lzo1x_worst_compress(PAGE_SIZE);
- lzo_ctx->dst = kmalloc(lzo_ctx->dst_len, GFP_KERNEL);
- if (!lzo_ctx->dst) {
- lzo_ctx->dst_len = 0;
- return -ENOMEM;
- }
-
- ret = regcache_lzo_compress(lzo_ctx);
- if (ret < 0)
- return ret;
- return 0;
-}
-
-static int regcache_lzo_decompress_cache_block(struct regmap *map,
- struct regcache_lzo_ctx *lzo_ctx)
-{
- int ret;
-
- lzo_ctx->dst_len = lzo_ctx->decompressed_size;
- lzo_ctx->dst = kmalloc(lzo_ctx->dst_len, GFP_KERNEL);
- if (!lzo_ctx->dst) {
- lzo_ctx->dst_len = 0;
- return -ENOMEM;
- }
-
- ret = regcache_lzo_decompress(lzo_ctx);
- if (ret < 0)
- return ret;
- return 0;
-}
-
-static inline int regcache_lzo_get_blkindex(struct regmap *map,
- unsigned int reg)
-{
- return (reg * map->cache_word_size) /
- DIV_ROUND_UP(map->cache_size_raw,
- regcache_lzo_block_count(map));
-}
-
-static inline int regcache_lzo_get_blkpos(struct regmap *map,
- unsigned int reg)
-{
- return reg % (DIV_ROUND_UP(map->cache_size_raw,
- regcache_lzo_block_count(map)) /
- map->cache_word_size);
-}
-
-static inline int regcache_lzo_get_blksize(struct regmap *map)
-{
- return DIV_ROUND_UP(map->cache_size_raw,
- regcache_lzo_block_count(map));
-}
-
-static int regcache_lzo_init(struct regmap *map)
-{
- struct regcache_lzo_ctx **lzo_blocks;
- size_t bmp_size;
- int ret, i, blksize, blkcount;
- const char *p, *end;
- unsigned long *sync_bmp;
-
- ret = 0;
-
- blkcount = regcache_lzo_block_count(map);
- map->cache = kzalloc(blkcount * sizeof *lzo_blocks,
- GFP_KERNEL);
- if (!map->cache)
- return -ENOMEM;
- lzo_blocks = map->cache;
-
- /*
- * allocate a bitmap to be used when syncing the cache with
- * the hardware. Each time a register is modified, the corresponding
- * bit is set in the bitmap, so we know that we have to sync
- * that register.
- */
- bmp_size = map->num_reg_defaults_raw;
- sync_bmp = kmalloc(BITS_TO_LONGS(bmp_size) * sizeof(long),
- GFP_KERNEL);
- if (!sync_bmp) {
- ret = -ENOMEM;
- goto err;
- }
- bitmap_zero(sync_bmp, bmp_size);
-
- /* allocate the lzo blocks and initialize them */
- for (i = 0; i < blkcount; i++) {
- lzo_blocks[i] = kzalloc(sizeof **lzo_blocks,
- GFP_KERNEL);
- if (!lzo_blocks[i]) {
- kfree(sync_bmp);
- ret = -ENOMEM;
- goto err;
- }
- lzo_blocks[i]->sync_bmp = sync_bmp;
- lzo_blocks[i]->sync_bmp_nbits = bmp_size;
- /* alloc the working space for the compressed block */
- ret = regcache_lzo_prepare(lzo_blocks[i]);
- if (ret < 0)
- goto err;
- }
-
- blksize = regcache_lzo_get_blksize(map);
- p = map->reg_defaults_raw;
- end = map->reg_defaults_raw + map->cache_size_raw;
- /* compress the register map and fill the lzo blocks */
- for (i = 0; i < blkcount; i++, p += blksize) {
- lzo_blocks[i]->src = p;
- if (p + blksize > end)
- lzo_blocks[i]->src_len = end - p;
- else
- lzo_blocks[i]->src_len = blksize;
- ret = regcache_lzo_compress_cache_block(map,
- lzo_blocks[i]);
- if (ret < 0)
- goto err;
- lzo_blocks[i]->decompressed_size =
- lzo_blocks[i]->src_len;
- }
-
- return 0;
-err:
- regcache_lzo_exit(map);
- return ret;
-}
-
-static int regcache_lzo_exit(struct regmap *map)
-{
- struct regcache_lzo_ctx **lzo_blocks;
- int i, blkcount;
-
- lzo_blocks = map->cache;
- if (!lzo_blocks)
- return 0;
-
- blkcount = regcache_lzo_block_count(map);
- /*
- * the pointer to the bitmap used for syncing the cache
- * is shared amongst all lzo_blocks. Ensure it is freed
- * only once.
- */
- if (lzo_blocks[0])
- kfree(lzo_blocks[0]->sync_bmp);
- for (i = 0; i < blkcount; i++) {
- if (lzo_blocks[i]) {
- kfree(lzo_blocks[i]->wmem);
- kfree(lzo_blocks[i]->dst);
- }
- /* each lzo_block is a pointer returned by kmalloc or NULL */
- kfree(lzo_blocks[i]);
- }
- kfree(lzo_blocks);
- map->cache = NULL;
- return 0;
-}
-
-static int regcache_lzo_read(struct regmap *map,
- unsigned int reg, unsigned int *value)
-{
- struct regcache_lzo_ctx *lzo_block, **lzo_blocks;
- int ret, blkindex, blkpos;
- size_t blksize, tmp_dst_len;
- void *tmp_dst;
-
- /* index of the compressed lzo block */
- blkindex = regcache_lzo_get_blkindex(map, reg);
- /* register index within the decompressed block */
- blkpos = regcache_lzo_get_blkpos(map, reg);
- /* size of the compressed block */
- blksize = regcache_lzo_get_blksize(map);
- lzo_blocks = map->cache;
- lzo_block = lzo_blocks[blkindex];
-
- /* save the pointer and length of the compressed block */
- tmp_dst = lzo_block->dst;
- tmp_dst_len = lzo_block->dst_len;
-
- /* prepare the source to be the compressed block */
- lzo_block->src = lzo_block->dst;
- lzo_block->src_len = lzo_block->dst_len;
-
- /* decompress the block */
- ret = regcache_lzo_decompress_cache_block(map, lzo_block);
- if (ret >= 0)
- /* fetch the value from the cache */
- *value = regcache_get_val(lzo_block->dst, blkpos,
- map->cache_word_size);
-
- kfree(lzo_block->dst);
- /* restore the pointer and length of the compressed block */
- lzo_block->dst = tmp_dst;
- lzo_block->dst_len = tmp_dst_len;
-
- return ret;
-}
-
-static int regcache_lzo_write(struct regmap *map,
- unsigned int reg, unsigned int value)
-{
- struct regcache_lzo_ctx *lzo_block, **lzo_blocks;
- int ret, blkindex, blkpos;
- size_t blksize, tmp_dst_len;
- void *tmp_dst;
-
- /* index of the compressed lzo block */
- blkindex = regcache_lzo_get_blkindex(map, reg);
- /* register index within the decompressed block */
- blkpos = regcache_lzo_get_blkpos(map, reg);
- /* size of the compressed block */
- blksize = regcache_lzo_get_blksize(map);
- lzo_blocks = map->cache;
- lzo_block = lzo_blocks[blkindex];
-
- /* save the pointer and length of the compressed block */
- tmp_dst = lzo_block->dst;
- tmp_dst_len = lzo_block->dst_len;
-
- /* prepare the source to be the compressed block */
- lzo_block->src = lzo_block->dst;
- lzo_block->src_len = lzo_block->dst_len;
-
- /* decompress the block */
- ret = regcache_lzo_decompress_cache_block(map, lzo_block);
- if (ret < 0) {
- kfree(lzo_block->dst);
- goto out;
- }
-
- /* write the new value to the cache */
- if (regcache_set_val(lzo_block->dst, blkpos, value,
- map->cache_word_size)) {
- kfree(lzo_block->dst);
- goto out;
- }
-
- /* prepare the source to be the decompressed block */
- lzo_block->src = lzo_block->dst;
- lzo_block->src_len = lzo_block->dst_len;
-
- /* compress the block */
- ret = regcache_lzo_compress_cache_block(map, lzo_block);
- if (ret < 0) {
- kfree(lzo_block->dst);
- kfree(lzo_block->src);
- goto out;
- }
-
- /* set the bit so we know we have to sync this register */
- set_bit(reg, lzo_block->sync_bmp);
- kfree(tmp_dst);
- kfree(lzo_block->src);
- return 0;
-out:
- lzo_block->dst = tmp_dst;
- lzo_block->dst_len = tmp_dst_len;
- return ret;
-}
-
-static int regcache_lzo_sync(struct regmap *map, unsigned int min,
- unsigned int max)
-{
- struct regcache_lzo_ctx **lzo_blocks;
- unsigned int val;
- int i;
- int ret;
-
- lzo_blocks = map->cache;
- i = min;
- for_each_set_bit_from(i, lzo_blocks[0]->sync_bmp,
- lzo_blocks[0]->sync_bmp_nbits) {
- if (i > max)
- continue;
-
- ret = regcache_read(map, i, &val);
- if (ret)
- return ret;
-
- /* Is this the hardware default? If so skip. */
- ret = regcache_lookup_reg(map, i);
- if (ret > 0 && val == map->reg_defaults[ret].def)
- continue;
-
- map->cache_bypass = 1;
- ret = _regmap_write(map, i, val);
- map->cache_bypass = 0;
- if (ret)
- return ret;
- dev_dbg(map->dev, "Synced register %#x, value %#x\n",
- i, val);
- }
-
- return 0;
-}
-
-struct regcache_ops regcache_lzo_ops = {
- .type = REGCACHE_COMPRESSED,
- .name = "lzo",
- .init = regcache_lzo_init,
- .exit = regcache_lzo_exit,
- .read = regcache_lzo_read,
- .write = regcache_lzo_write,
- .sync = regcache_lzo_sync
-};
diff --git a/ANDROID_3.4.5/drivers/base/regmap/regcache-rbtree.c b/ANDROID_3.4.5/drivers/base/regmap/regcache-rbtree.c
deleted file mode 100644
index 92b779ee..00000000
--- a/ANDROID_3.4.5/drivers/base/regmap/regcache-rbtree.c
+++ /dev/null
@@ -1,430 +0,0 @@
-/*
- * Register cache access API - rbtree caching support
- *
- * Copyright 2011 Wolfson Microelectronics plc
- *
- * Author: Dimitris Papastamos <dp@opensource.wolfsonmicro.com>
- *
- * This program is free software; you can redistribute it and/or modify
- * it under the terms of the GNU General Public License version 2 as
- * published by the Free Software Foundation.
- */
-
-#include <linux/slab.h>
-#include <linux/device.h>
-#include <linux/debugfs.h>
-#include <linux/rbtree.h>
-#include <linux/seq_file.h>
-
-#include "internal.h"
-
-static int regcache_rbtree_write(struct regmap *map, unsigned int reg,
- unsigned int value);
-static int regcache_rbtree_exit(struct regmap *map);
-
-struct regcache_rbtree_node {
- /* the actual rbtree node holding this block */
- struct rb_node node;
- /* base register handled by this block */
- unsigned int base_reg;
- /* block of adjacent registers */
- void *block;
- /* number of registers available in the block */
- unsigned int blklen;
-} __attribute__ ((packed));
-
-struct regcache_rbtree_ctx {
- struct rb_root root;
- struct regcache_rbtree_node *cached_rbnode;
-};
-
-static inline void regcache_rbtree_get_base_top_reg(
- struct regcache_rbtree_node *rbnode,
- unsigned int *base, unsigned int *top)
-{
- *base = rbnode->base_reg;
- *top = rbnode->base_reg + rbnode->blklen - 1;
-}
-
-static unsigned int regcache_rbtree_get_register(
- struct regcache_rbtree_node *rbnode, unsigned int idx,
- unsigned int word_size)
-{
- return regcache_get_val(rbnode->block, idx, word_size);
-}
-
-static void regcache_rbtree_set_register(struct regcache_rbtree_node *rbnode,
- unsigned int idx, unsigned int val,
- unsigned int word_size)
-{
- regcache_set_val(rbnode->block, idx, val, word_size);
-}
-
-static struct regcache_rbtree_node *regcache_rbtree_lookup(struct regmap *map,
- unsigned int reg)
-{
- struct regcache_rbtree_ctx *rbtree_ctx = map->cache;
- struct rb_node *node;
- struct regcache_rbtree_node *rbnode;
- unsigned int base_reg, top_reg;
-
- rbnode = rbtree_ctx->cached_rbnode;
- if (rbnode) {
- regcache_rbtree_get_base_top_reg(rbnode, &base_reg, &top_reg);
- if (reg >= base_reg && reg <= top_reg)
- return rbnode;
- }
-
- node = rbtree_ctx->root.rb_node;
- while (node) {
- rbnode = container_of(node, struct regcache_rbtree_node, node);
- regcache_rbtree_get_base_top_reg(rbnode, &base_reg, &top_reg);
- if (reg >= base_reg && reg <= top_reg) {
- rbtree_ctx->cached_rbnode = rbnode;
- return rbnode;
- } else if (reg > top_reg) {
- node = node->rb_right;
- } else if (reg < base_reg) {
- node = node->rb_left;
- }
- }
-
- return NULL;
-}
-
-static int regcache_rbtree_insert(struct rb_root *root,
- struct regcache_rbtree_node *rbnode)
-{
- struct rb_node **new, *parent;
- struct regcache_rbtree_node *rbnode_tmp;
- unsigned int base_reg_tmp, top_reg_tmp;
- unsigned int base_reg;
-
- parent = NULL;
- new = &root->rb_node;
- while (*new) {
- rbnode_tmp = container_of(*new, struct regcache_rbtree_node,
- node);
- /* base and top registers of the current rbnode */
- regcache_rbtree_get_base_top_reg(rbnode_tmp, &base_reg_tmp,
- &top_reg_tmp);
- /* base register of the rbnode to be added */
- base_reg = rbnode->base_reg;
- parent = *new;
- /* if this register has already been inserted, just return */
- if (base_reg >= base_reg_tmp &&
- base_reg <= top_reg_tmp)
- return 0;
- else if (base_reg > top_reg_tmp)
- new = &((*new)->rb_right);
- else if (base_reg < base_reg_tmp)
- new = &((*new)->rb_left);
- }
-
- /* insert the node into the rbtree */
- rb_link_node(&rbnode->node, parent, new);
- rb_insert_color(&rbnode->node, root);
-
- return 1;
-}
-
-#ifdef CONFIG_DEBUG_FS
-static int rbtree_show(struct seq_file *s, void *ignored)
-{
- struct regmap *map = s->private;
- struct regcache_rbtree_ctx *rbtree_ctx = map->cache;
- struct regcache_rbtree_node *n;
- struct rb_node *node;
- unsigned int base, top;
- int nodes = 0;
- int registers = 0;
- int average;
-
- mutex_lock(&map->lock);
-
- for (node = rb_first(&rbtree_ctx->root); node != NULL;
- node = rb_next(node)) {
- n = container_of(node, struct regcache_rbtree_node, node);
-
- regcache_rbtree_get_base_top_reg(n, &base, &top);
- seq_printf(s, "%x-%x (%d)\n", base, top, top - base + 1);
-
- nodes++;
- registers += top - base + 1;
- }
-
- if (nodes)
- average = registers / nodes;
- else
- average = 0;
-
- seq_printf(s, "%d nodes, %d registers, average %d registers\n",
- nodes, registers, average);
-
- mutex_unlock(&map->lock);
-
- return 0;
-}
-
-static int rbtree_open(struct inode *inode, struct file *file)
-{
- return single_open(file, rbtree_show, inode->i_private);
-}
-
-static const struct file_operations rbtree_fops = {
- .open = rbtree_open,
- .read = seq_read,
- .llseek = seq_lseek,
- .release = single_release,
-};
-
-static void rbtree_debugfs_init(struct regmap *map)
-{
- debugfs_create_file("rbtree", 0400, map->debugfs, map, &rbtree_fops);
-}
-#else
-static void rbtree_debugfs_init(struct regmap *map)
-{
-}
-#endif
-
-static int regcache_rbtree_init(struct regmap *map)
-{
- struct regcache_rbtree_ctx *rbtree_ctx;
- int i;
- int ret;
-
- map->cache = kmalloc(sizeof *rbtree_ctx, GFP_KERNEL);
- if (!map->cache)
- return -ENOMEM;
-
- rbtree_ctx = map->cache;
- rbtree_ctx->root = RB_ROOT;
- rbtree_ctx->cached_rbnode = NULL;
-
- for (i = 0; i < map->num_reg_defaults; i++) {
- ret = regcache_rbtree_write(map,
- map->reg_defaults[i].reg,
- map->reg_defaults[i].def);
- if (ret)
- goto err;
- }
-
- rbtree_debugfs_init(map);
-
- return 0;
-
-err:
- regcache_rbtree_exit(map);
- return ret;
-}
-
-static int regcache_rbtree_exit(struct regmap *map)
-{
- struct rb_node *next;
- struct regcache_rbtree_ctx *rbtree_ctx;
- struct regcache_rbtree_node *rbtree_node;
-
- /* if we've already been called then just return */
- rbtree_ctx = map->cache;
- if (!rbtree_ctx)
- return 0;
-
- /* free up the rbtree */
- next = rb_first(&rbtree_ctx->root);
- while (next) {
- rbtree_node = rb_entry(next, struct regcache_rbtree_node, node);
- next = rb_next(&rbtree_node->node);
- rb_erase(&rbtree_node->node, &rbtree_ctx->root);
- kfree(rbtree_node->block);
- kfree(rbtree_node);
- }
-
- /* release the resources */
- kfree(map->cache);
- map->cache = NULL;
-
- return 0;
-}
-
-static int regcache_rbtree_read(struct regmap *map,
- unsigned int reg, unsigned int *value)
-{
- struct regcache_rbtree_node *rbnode;
- unsigned int reg_tmp;
-
- rbnode = regcache_rbtree_lookup(map, reg);
- if (rbnode) {
- reg_tmp = reg - rbnode->base_reg;
- *value = regcache_rbtree_get_register(rbnode, reg_tmp,
- map->cache_word_size);
- } else {
- return -ENOENT;
- }
-
- return 0;
-}
-
-
-static int regcache_rbtree_insert_to_block(struct regcache_rbtree_node *rbnode,
- unsigned int pos, unsigned int reg,
- unsigned int value, unsigned int word_size)
-{
- u8 *blk;
-
- blk = krealloc(rbnode->block,
- (rbnode->blklen + 1) * word_size, GFP_KERNEL);
- if (!blk)
- return -ENOMEM;
-
- /* insert the register value in the correct place in the rbnode block */
- memmove(blk + (pos + 1) * word_size,
- blk + pos * word_size,
- (rbnode->blklen - pos) * word_size);
-
- /* update the rbnode block, its size and the base register */
- rbnode->block = blk;
- rbnode->blklen++;
- if (!pos)
- rbnode->base_reg = reg;
-
- regcache_rbtree_set_register(rbnode, pos, value, word_size);
- return 0;
-}
-
-static int regcache_rbtree_write(struct regmap *map, unsigned int reg,
- unsigned int value)
-{
- struct regcache_rbtree_ctx *rbtree_ctx;
- struct regcache_rbtree_node *rbnode, *rbnode_tmp;
- struct rb_node *node;
- unsigned int val;
- unsigned int reg_tmp;
- unsigned int pos;
- int i;
- int ret;
-
- rbtree_ctx = map->cache;
- /* if we can't locate it in the cached rbnode we'll have
- * to traverse the rbtree looking for it.
- */
- rbnode = regcache_rbtree_lookup(map, reg);
- if (rbnode) {
- reg_tmp = reg - rbnode->base_reg;
- val = regcache_rbtree_get_register(rbnode, reg_tmp,
- map->cache_word_size);
- if (val == value)
- return 0;
- regcache_rbtree_set_register(rbnode, reg_tmp, value,
- map->cache_word_size);
- } else {
- /* look for an adjacent register to the one we are about to add */
- for (node = rb_first(&rbtree_ctx->root); node;
- node = rb_next(node)) {
- rbnode_tmp = rb_entry(node, struct regcache_rbtree_node, node);
- for (i = 0; i < rbnode_tmp->blklen; i++) {
- reg_tmp = rbnode_tmp->base_reg + i;
- if (abs(reg_tmp - reg) != 1)
- continue;
- /* decide where in the block to place our register */
- if (reg_tmp + 1 == reg)
- pos = i + 1;
- else
- pos = i;
- ret = regcache_rbtree_insert_to_block(rbnode_tmp, pos,
- reg, value,
- map->cache_word_size);
- if (ret)
- return ret;
- rbtree_ctx->cached_rbnode = rbnode_tmp;
- return 0;
- }
- }
- /* we did not manage to find a place to insert it in an existing
- * block so create a new rbnode with a single register in its block.
- * This block will get populated further if any other adjacent
- * registers get modified in the future.
- */
- rbnode = kzalloc(sizeof *rbnode, GFP_KERNEL);
- if (!rbnode)
- return -ENOMEM;
- rbnode->blklen = 1;
- rbnode->base_reg = reg;
- rbnode->block = kmalloc(rbnode->blklen * map->cache_word_size,
- GFP_KERNEL);
- if (!rbnode->block) {
- kfree(rbnode);
- return -ENOMEM;
- }
- regcache_rbtree_set_register(rbnode, 0, value, map->cache_word_size);
- regcache_rbtree_insert(&rbtree_ctx->root, rbnode);
- rbtree_ctx->cached_rbnode = rbnode;
- }
-
- return 0;
-}
-
-static int regcache_rbtree_sync(struct regmap *map, unsigned int min,
- unsigned int max)
-{
- struct regcache_rbtree_ctx *rbtree_ctx;
- struct rb_node *node;
- struct regcache_rbtree_node *rbnode;
- unsigned int regtmp;
- unsigned int val;
- int ret;
- int i, base, end;
-
- rbtree_ctx = map->cache;
- for (node = rb_first(&rbtree_ctx->root); node; node = rb_next(node)) {
- rbnode = rb_entry(node, struct regcache_rbtree_node, node);
-
- if (rbnode->base_reg < min)
- continue;
- if (rbnode->base_reg > max)
- break;
- if (rbnode->base_reg + rbnode->blklen < min)
- continue;
-
- if (min > rbnode->base_reg)
- base = min - rbnode->base_reg;
- else
- base = 0;
-
- if (max < rbnode->base_reg + rbnode->blklen)
- end = rbnode->base_reg + rbnode->blklen - max;
- else
- end = rbnode->blklen;
-
- for (i = base; i < end; i++) {
- regtmp = rbnode->base_reg + i;
- val = regcache_rbtree_get_register(rbnode, i,
- map->cache_word_size);
-
- /* Is this the hardware default? If so skip. */
- ret = regcache_lookup_reg(map, regtmp);
- if (ret >= 0 && val == map->reg_defaults[ret].def)
- continue;
-
- map->cache_bypass = 1;
- ret = _regmap_write(map, regtmp, val);
- map->cache_bypass = 0;
- if (ret)
- return ret;
- dev_dbg(map->dev, "Synced register %#x, value %#x\n",
- regtmp, val);
- }
- }
-
- return 0;
-}
-
-struct regcache_ops regcache_rbtree_ops = {
- .type = REGCACHE_RBTREE,
- .name = "rbtree",
- .init = regcache_rbtree_init,
- .exit = regcache_rbtree_exit,
- .read = regcache_rbtree_read,
- .write = regcache_rbtree_write,
- .sync = regcache_rbtree_sync
-};
diff --git a/ANDROID_3.4.5/drivers/base/regmap/regcache.c b/ANDROID_3.4.5/drivers/base/regmap/regcache.c
deleted file mode 100644
index 74b69095..00000000
--- a/ANDROID_3.4.5/drivers/base/regmap/regcache.c
+++ /dev/null
@@ -1,491 +0,0 @@
-/*
- * Register cache access API
- *
- * Copyright 2011 Wolfson Microelectronics plc
- *
- * Author: Dimitris Papastamos <dp@opensource.wolfsonmicro.com>
- *
- * This program is free software; you can redistribute it and/or modify
- * it under the terms of the GNU General Public License version 2 as
- * published by the Free Software Foundation.
- */
-
-#include <linux/slab.h>
-#include <linux/export.h>
-#include <linux/device.h>
-#include <trace/events/regmap.h>
-#include <linux/bsearch.h>
-#include <linux/sort.h>
-
-#include "internal.h"
-
-static const struct regcache_ops *cache_types[] = {
- &regcache_rbtree_ops,
- &regcache_lzo_ops,
-};
-
-static int regcache_hw_init(struct regmap *map)
-{
- int i, j;
- int ret;
- int count;
- unsigned int val;
- void *tmp_buf;
-
- if (!map->num_reg_defaults_raw)
- return -EINVAL;
-
- if (!map->reg_defaults_raw) {
- u32 cache_bypass = map->cache_bypass;
- dev_warn(map->dev, "No cache defaults, reading back from HW\n");
-
- /* Bypass the cache access till data read from HW*/
- map->cache_bypass = 1;
- tmp_buf = kmalloc(map->cache_size_raw, GFP_KERNEL);
- if (!tmp_buf)
- return -EINVAL;
- ret = regmap_bulk_read(map, 0, tmp_buf,
- map->num_reg_defaults_raw);
- map->cache_bypass = cache_bypass;
- if (ret < 0) {
- kfree(tmp_buf);
- return ret;
- }
- map->reg_defaults_raw = tmp_buf;
- map->cache_free = 1;
- }
-
- /* calculate the size of reg_defaults */
- for (count = 0, i = 0; i < map->num_reg_defaults_raw; i++) {
- val = regcache_get_val(map->reg_defaults_raw,
- i, map->cache_word_size);
- if (regmap_volatile(map, i))
- continue;
- count++;
- }
-
- map->reg_defaults = kmalloc(count * sizeof(struct reg_default),
- GFP_KERNEL);
- if (!map->reg_defaults) {
- ret = -ENOMEM;
- goto err_free;
- }
-
- /* fill the reg_defaults */
- map->num_reg_defaults = count;
- for (i = 0, j = 0; i < map->num_reg_defaults_raw; i++) {
- val = regcache_get_val(map->reg_defaults_raw,
- i, map->cache_word_size);
- if (regmap_volatile(map, i))
- continue;
- map->reg_defaults[j].reg = i;
- map->reg_defaults[j].def = val;
- j++;
- }
-
- return 0;
-
-err_free:
- if (map->cache_free)
- kfree(map->reg_defaults_raw);
-
- return ret;
-}
-
-int regcache_init(struct regmap *map, const struct regmap_config *config)
-{
- int ret;
- int i;
- void *tmp_buf;
-
- if (map->cache_type == REGCACHE_NONE) {
- map->cache_bypass = true;
- return 0;
- }
-
- for (i = 0; i < ARRAY_SIZE(cache_types); i++)
- if (cache_types[i]->type == map->cache_type)
- break;
-
- if (i == ARRAY_SIZE(cache_types)) {
- dev_err(map->dev, "Could not match compress type: %d\n",
- map->cache_type);
- return -EINVAL;
- }
-
- map->num_reg_defaults = config->num_reg_defaults;
- map->num_reg_defaults_raw = config->num_reg_defaults_raw;
- map->reg_defaults_raw = config->reg_defaults_raw;
- map->cache_word_size = DIV_ROUND_UP(config->val_bits, 8);
- map->cache_size_raw = map->cache_word_size * config->num_reg_defaults_raw;
-
- map->cache = NULL;
- map->cache_ops = cache_types[i];
-
- if (!map->cache_ops->read ||
- !map->cache_ops->write ||
- !map->cache_ops->name)
- return -EINVAL;
-
- /* We still need to ensure that the reg_defaults
- * won't vanish from under us. We'll need to make
- * a copy of it.
- */
- if (config->reg_defaults) {
- if (!map->num_reg_defaults)
- return -EINVAL;
- tmp_buf = kmemdup(config->reg_defaults, map->num_reg_defaults *
- sizeof(struct reg_default), GFP_KERNEL);
- if (!tmp_buf)
- return -ENOMEM;
- map->reg_defaults = tmp_buf;
- } else if (map->num_reg_defaults_raw) {
- /* Some devices such as PMICs don't have cache defaults,
- * we cope with this by reading back the HW registers and
- * crafting the cache defaults by hand.
- */
- ret = regcache_hw_init(map);
- if (ret < 0)
- return ret;
- }
-
- if (!map->max_register)
- map->max_register = map->num_reg_defaults_raw;
-
- if (map->cache_ops->init) {
- dev_dbg(map->dev, "Initializing %s cache\n",
- map->cache_ops->name);
- ret = map->cache_ops->init(map);
- if (ret)
- goto err_free;
- }
- return 0;
-
-err_free:
- kfree(map->reg_defaults);
- if (map->cache_free)
- kfree(map->reg_defaults_raw);
-
- return ret;
-}
-
-void regcache_exit(struct regmap *map)
-{
- if (map->cache_type == REGCACHE_NONE)
- return;
-
- BUG_ON(!map->cache_ops);
-
- kfree(map->reg_defaults);
- if (map->cache_free)
- kfree(map->reg_defaults_raw);
-
- if (map->cache_ops->exit) {
- dev_dbg(map->dev, "Destroying %s cache\n",
- map->cache_ops->name);
- map->cache_ops->exit(map);
- }
-}
-
-/**
- * regcache_read: Fetch the value of a given register from the cache.
- *
- * @map: map to configure.
- * @reg: The register index.
- * @value: The value to be returned.
- *
- * Return a negative value on failure, 0 on success.
- */
-int regcache_read(struct regmap *map,
- unsigned int reg, unsigned int *value)
-{
- int ret;
-
- if (map->cache_type == REGCACHE_NONE)
- return -ENOSYS;
-
- BUG_ON(!map->cache_ops);
-
- if (!regmap_volatile(map, reg)) {
- ret = map->cache_ops->read(map, reg, value);
-
- if (ret == 0)
- trace_regmap_reg_read_cache(map->dev, reg, *value);
-
- return ret;
- }
-
- return -EINVAL;
-}
-
-/**
- * regcache_write: Set the value of a given register in the cache.
- *
- * @map: map to configure.
- * @reg: The register index.
- * @value: The new register value.
- *
- * Return a negative value on failure, 0 on success.
- */
-int regcache_write(struct regmap *map,
- unsigned int reg, unsigned int value)
-{
- if (map->cache_type == REGCACHE_NONE)
- return 0;
-
- BUG_ON(!map->cache_ops);
-
- if (!regmap_writeable(map, reg))
- return -EIO;
-
- if (!regmap_volatile(map, reg))
- return map->cache_ops->write(map, reg, value);
-
- return 0;
-}
-
-/**
- * regcache_sync: Sync the register cache with the hardware.
- *
- * @map: map to configure.
- *
- * Any registers that should not be synced should be marked as
- * volatile. In general drivers can choose not to use the provided
- * syncing functionality if they so require.
- *
- * Return a negative value on failure, 0 on success.
- */
-int regcache_sync(struct regmap *map)
-{
- int ret = 0;
- unsigned int i;
- const char *name;
- unsigned int bypass;
-
- BUG_ON(!map->cache_ops || !map->cache_ops->sync);
-
- mutex_lock(&map->lock);
- /* Remember the initial bypass state */
- bypass = map->cache_bypass;
- dev_dbg(map->dev, "Syncing %s cache\n",
- map->cache_ops->name);
- name = map->cache_ops->name;
- trace_regcache_sync(map->dev, name, "start");
-
- if (!map->cache_dirty)
- goto out;
-
- /* Apply any patch first */
- map->cache_bypass = 1;
- for (i = 0; i < map->patch_regs; i++) {
- ret = _regmap_write(map, map->patch[i].reg, map->patch[i].def);
- if (ret != 0) {
- dev_err(map->dev, "Failed to write %x = %x: %d\n",
- map->patch[i].reg, map->patch[i].def, ret);
- goto out;
- }
- }
- map->cache_bypass = 0;
-
- ret = map->cache_ops->sync(map, 0, map->max_register);
-
- if (ret == 0)
- map->cache_dirty = false;
-
-out:
- trace_regcache_sync(map->dev, name, "stop");
- /* Restore the bypass state */
- map->cache_bypass = bypass;
- mutex_unlock(&map->lock);
-
- return ret;
-}
-EXPORT_SYMBOL_GPL(regcache_sync);
-
-/**
- * regcache_sync_region: Sync part of the register cache with the hardware.
- *
- * @map: map to sync.
- * @min: first register to sync
- * @max: last register to sync
- *
- * Write all non-default register values in the specified region to
- * the hardware.
- *
- * Return a negative value on failure, 0 on success.
- */
-int regcache_sync_region(struct regmap *map, unsigned int min,
- unsigned int max)
-{
- int ret = 0;
- const char *name;
- unsigned int bypass;
-
- BUG_ON(!map->cache_ops || !map->cache_ops->sync);
-
- mutex_lock(&map->lock);
-
- /* Remember the initial bypass state */
- bypass = map->cache_bypass;
-
- name = map->cache_ops->name;
- dev_dbg(map->dev, "Syncing %s cache from %d-%d\n", name, min, max);
-
- trace_regcache_sync(map->dev, name, "start region");
-
- if (!map->cache_dirty)
- goto out;
-
- ret = map->cache_ops->sync(map, min, max);
-
-out:
- trace_regcache_sync(map->dev, name, "stop region");
- /* Restore the bypass state */
- map->cache_bypass = bypass;
- mutex_unlock(&map->lock);
-
- return ret;
-}
-EXPORT_SYMBOL_GPL(regcache_sync_region);
-
-/**
- * regcache_cache_only: Put a register map into cache only mode
- *
- * @map: map to configure
- * @cache_only: flag if changes should be written to the hardware
- *
- * When a register map is marked as cache only writes to the register
- * map API will only update the register cache, they will not cause
- * any hardware changes. This is useful for allowing portions of
- * drivers to act as though the device were functioning as normal when
- * it is disabled for power saving reasons.
- */
-void regcache_cache_only(struct regmap *map, bool enable)
-{
- mutex_lock(&map->lock);
- WARN_ON(map->cache_bypass && enable);
- map->cache_only = enable;
- trace_regmap_cache_only(map->dev, enable);
- mutex_unlock(&map->lock);
-}
-EXPORT_SYMBOL_GPL(regcache_cache_only);
-
-/**
- * regcache_mark_dirty: Mark the register cache as dirty
- *
- * @map: map to mark
- *
- * Mark the register cache as dirty, for example due to the device
- * having been powered down for suspend. If the cache is not marked
- * as dirty then the cache sync will be suppressed.
- */
-void regcache_mark_dirty(struct regmap *map)
-{
- mutex_lock(&map->lock);
- map->cache_dirty = true;
- mutex_unlock(&map->lock);
-}
-EXPORT_SYMBOL_GPL(regcache_mark_dirty);
-
-/**
- * regcache_cache_bypass: Put a register map into cache bypass mode
- *
- * @map: map to configure
- * @cache_bypass: flag if changes should not be written to the hardware
- *
- * When a register map is marked with the cache bypass option, writes
- * to the register map API will only update the hardware and not the
- * the cache directly. This is useful when syncing the cache back to
- * the hardware.
- */
-void regcache_cache_bypass(struct regmap *map, bool enable)
-{
- mutex_lock(&map->lock);
- WARN_ON(map->cache_only && enable);
- map->cache_bypass = enable;
- trace_regmap_cache_bypass(map->dev, enable);
- mutex_unlock(&map->lock);
-}
-EXPORT_SYMBOL_GPL(regcache_cache_bypass);
-
-bool regcache_set_val(void *base, unsigned int idx,
- unsigned int val, unsigned int word_size)
-{
- switch (word_size) {
- case 1: {
- u8 *cache = base;
- if (cache[idx] == val)
- return true;
- cache[idx] = val;
- break;
- }
- case 2: {
- u16 *cache = base;
- if (cache[idx] == val)
- return true;
- cache[idx] = val;
- break;
- }
- case 4: {
- u32 *cache = base;
- if (cache[idx] == val)
- return true;
- cache[idx] = val;
- break;
- }
- default:
- BUG();
- }
- return false;
-}
-
-unsigned int regcache_get_val(const void *base, unsigned int idx,
- unsigned int word_size)
-{
- if (!base)
- return -EINVAL;
-
- switch (word_size) {
- case 1: {
- const u8 *cache = base;
- return cache[idx];
- }
- case 2: {
- const u16 *cache = base;
- return cache[idx];
- }
- case 4: {
- const u32 *cache = base;
- return cache[idx];
- }
- default:
- BUG();
- }
- /* unreachable */
- return -1;
-}
-
-static int regcache_default_cmp(const void *a, const void *b)
-{
- const struct reg_default *_a = a;
- const struct reg_default *_b = b;
-
- return _a->reg - _b->reg;
-}
-
-int regcache_lookup_reg(struct regmap *map, unsigned int reg)
-{
- struct reg_default key;
- struct reg_default *r;
-
- key.reg = reg;
- key.def = 0;
-
- r = bsearch(&key, map->reg_defaults, map->num_reg_defaults,
- sizeof(struct reg_default), regcache_default_cmp);
-
- if (r)
- return r - map->reg_defaults;
- else
- return -ENOENT;
-}
diff --git a/ANDROID_3.4.5/drivers/base/regmap/regmap-debugfs.c b/ANDROID_3.4.5/drivers/base/regmap/regmap-debugfs.c
deleted file mode 100644
index 8ee03493..00000000
--- a/ANDROID_3.4.5/drivers/base/regmap/regmap-debugfs.c
+++ /dev/null
@@ -1,286 +0,0 @@
-/*
- * Register map access API - debugfs
- *
- * Copyright 2011 Wolfson Microelectronics plc
- *
- * Author: Mark Brown <broonie@opensource.wolfsonmicro.com>
- *
- * This program is free software; you can redistribute it and/or modify
- * it under the terms of the GNU General Public License version 2 as
- * published by the Free Software Foundation.
- */
-
-#include <linux/slab.h>
-#include <linux/mutex.h>
-#include <linux/debugfs.h>
-#include <linux/uaccess.h>
-#include <linux/device.h>
-
-#include "internal.h"
-
-static struct dentry *regmap_debugfs_root;
-
-/* Calculate the length of a fixed format */
-static size_t regmap_calc_reg_len(int max_val, char *buf, size_t buf_size)
-{
- snprintf(buf, buf_size, "%x", max_val);
- return strlen(buf);
-}
-
-static ssize_t regmap_name_read_file(struct file *file,
- char __user *user_buf, size_t count,
- loff_t *ppos)
-{
- struct regmap *map = file->private_data;
- int ret;
- char *buf;
-
- buf = kmalloc(PAGE_SIZE, GFP_KERNEL);
- if (!buf)
- return -ENOMEM;
-
- ret = snprintf(buf, PAGE_SIZE, "%s\n", map->dev->driver->name);
- if (ret < 0) {
- kfree(buf);
- return ret;
- }
-
- ret = simple_read_from_buffer(user_buf, count, ppos, buf, ret);
- kfree(buf);
- return ret;
-}
-
-static const struct file_operations regmap_name_fops = {
- .open = simple_open,
- .read = regmap_name_read_file,
- .llseek = default_llseek,
-};
-
-static ssize_t regmap_map_read_file(struct file *file, char __user *user_buf,
- size_t count, loff_t *ppos)
-{
- int reg_len, val_len, tot_len;
- size_t buf_pos = 0;
- loff_t p = 0;
- ssize_t ret;
- int i;
- struct regmap *map = file->private_data;
- char *buf;
- unsigned int val;
-
- if (*ppos < 0 || !count)
- return -EINVAL;
-
- buf = kmalloc(count, GFP_KERNEL);
- if (!buf)
- return -ENOMEM;
-
- /* Calculate the length of a fixed format */
- reg_len = regmap_calc_reg_len(map->max_register, buf, count);
- val_len = 2 * map->format.val_bytes;
- tot_len = reg_len + val_len + 3; /* : \n */
-
- for (i = 0; i < map->max_register + 1; i++) {
- if (!regmap_readable(map, i))
- continue;
-
- if (regmap_precious(map, i))
- continue;
-
- /* If we're in the region the user is trying to read */
- if (p >= *ppos) {
- /* ...but not beyond it */
- if (buf_pos >= count - 1 - tot_len)
- break;
-
- /* Format the register */
- snprintf(buf + buf_pos, count - buf_pos, "%.*x: ",
- reg_len, i);
- buf_pos += reg_len + 2;
-
- /* Format the value, write all X if we can't read */
- ret = regmap_read(map, i, &val);
- if (ret == 0)
- snprintf(buf + buf_pos, count - buf_pos,
- "%.*x", val_len, val);
- else
- memset(buf + buf_pos, 'X', val_len);
- buf_pos += 2 * map->format.val_bytes;
-
- buf[buf_pos++] = '\n';
- }
- p += tot_len;
- }
-
- ret = buf_pos;
-
- if (copy_to_user(user_buf, buf, buf_pos)) {
- ret = -EFAULT;
- goto out;
- }
-
- *ppos += buf_pos;
-
-out:
- kfree(buf);
- return ret;
-}
-
-#define REGMAP_ALLOW_WRITE_DEBUGFS
-#ifdef REGMAP_ALLOW_WRITE_DEBUGFS
-/*
- * This can be dangerous especially when we have clients such as
- * PMICs, therefore don't provide any real compile time configuration option
- * for this feature, people who want to use this will need to modify
- * the source code directly.
- */
-static ssize_t regmap_map_write_file(struct file *file,
- const char __user *user_buf,
- size_t count, loff_t *ppos)
-{
- char buf[32];
- size_t buf_size;
- char *start = buf;
- unsigned long reg, value;
- struct regmap *map = file->private_data;
-
- buf_size = min(count, (sizeof(buf)-1));
- if (copy_from_user(buf, user_buf, buf_size))
- return -EFAULT;
- buf[buf_size] = 0;
-
- while (*start == ' ')
- start++;
- reg = simple_strtoul(start, &start, 16);
- while (*start == ' ')
- start++;
- if (strict_strtoul(start, 16, &value))
- return -EINVAL;
-
- /* Userspace has been fiddling around behind the kernel's back */
- add_taint(TAINT_USER);
-
- regmap_write(map, reg, value);
- return buf_size;
-}
-#else
-#define regmap_map_write_file NULL
-#endif
-
-static const struct file_operations regmap_map_fops = {
- .open = simple_open,
- .read = regmap_map_read_file,
- .write = regmap_map_write_file,
- .llseek = default_llseek,
-};
-
-static ssize_t regmap_access_read_file(struct file *file,
- char __user *user_buf, size_t count,
- loff_t *ppos)
-{
- int reg_len, tot_len;
- size_t buf_pos = 0;
- loff_t p = 0;
- ssize_t ret;
- int i;
- struct regmap *map = file->private_data;
- char *buf;
-
- if (*ppos < 0 || !count)
- return -EINVAL;
-
- buf = kmalloc(count, GFP_KERNEL);
- if (!buf)
- return -ENOMEM;
-
- /* Calculate the length of a fixed format */
- reg_len = regmap_calc_reg_len(map->max_register, buf, count);
- tot_len = reg_len + 10; /* ': R W V P\n' */
-
- for (i = 0; i < map->max_register + 1; i++) {
- /* Ignore registers which are neither readable nor writable */
- if (!regmap_readable(map, i) && !regmap_writeable(map, i))
- continue;
-
- /* If we're in the region the user is trying to read */
- if (p >= *ppos) {
- /* ...but not beyond it */
- if (buf_pos >= count - 1 - tot_len)
- break;
-
- /* Format the register */
- snprintf(buf + buf_pos, count - buf_pos,
- "%.*x: %c %c %c %c\n",
- reg_len, i,
- regmap_readable(map, i) ? 'y' : 'n',
- regmap_writeable(map, i) ? 'y' : 'n',
- regmap_volatile(map, i) ? 'y' : 'n',
- regmap_precious(map, i) ? 'y' : 'n');
-
- buf_pos += tot_len;
- }
- p += tot_len;
- }
-
- ret = buf_pos;
-
- if (copy_to_user(user_buf, buf, buf_pos)) {
- ret = -EFAULT;
- goto out;
- }
-
- *ppos += buf_pos;
-
-out:
- kfree(buf);
- return ret;
-}
-
-static const struct file_operations regmap_access_fops = {
- .open = simple_open,
- .read = regmap_access_read_file,
- .llseek = default_llseek,
-};
-
-void regmap_debugfs_init(struct regmap *map)
-{
- map->debugfs = debugfs_create_dir(dev_name(map->dev),
- regmap_debugfs_root);
- if (!map->debugfs) {
- dev_warn(map->dev, "Failed to create debugfs directory\n");
- return;
- }
-
- debugfs_create_file("name", 0400, map->debugfs,
- map, &regmap_name_fops);
-
- if (map->max_register) {
- debugfs_create_file("registers", 0400, map->debugfs,
- map, &regmap_map_fops);
- debugfs_create_file("access", 0400, map->debugfs,
- map, &regmap_access_fops);
- }
-
- if (map->cache_type) {
- debugfs_create_bool("cache_only", 0400, map->debugfs,
- &map->cache_only);
- debugfs_create_bool("cache_dirty", 0400, map->debugfs,
- &map->cache_dirty);
- debugfs_create_bool("cache_bypass", 0400, map->debugfs,
- &map->cache_bypass);
- }
-}
-
-void regmap_debugfs_exit(struct regmap *map)
-{
- debugfs_remove_recursive(map->debugfs);
-}
-
-void regmap_debugfs_initcall(void)
-{
- regmap_debugfs_root = debugfs_create_dir("regmap", NULL);
- if (!regmap_debugfs_root) {
- pr_warn("regmap: Failed to create debugfs root\n");
- return;
- }
-}
diff --git a/ANDROID_3.4.5/drivers/base/regmap/regmap-i2c.c b/ANDROID_3.4.5/drivers/base/regmap/regmap-i2c.c
deleted file mode 100644
index 9a3a8c56..00000000
--- a/ANDROID_3.4.5/drivers/base/regmap/regmap-i2c.c
+++ /dev/null
@@ -1,131 +0,0 @@
-/*
- * Register map access API - I2C support
- *
- * Copyright 2011 Wolfson Microelectronics plc
- *
- * Author: Mark Brown <broonie@opensource.wolfsonmicro.com>
- *
- * This program is free software; you can redistribute it and/or modify
- * it under the terms of the GNU General Public License version 2 as
- * published by the Free Software Foundation.
- */
-
-#include <linux/regmap.h>
-#include <linux/i2c.h>
-#include <linux/module.h>
-#include <linux/init.h>
-
-static int regmap_i2c_write(struct device *dev, const void *data, size_t count)
-{
- struct i2c_client *i2c = to_i2c_client(dev);
- int ret;
-
- ret = i2c_master_send(i2c, data, count);
- if (ret == count)
- return 0;
- else if (ret < 0)
- return ret;
- else
- return -EIO;
-}
-
-static int regmap_i2c_gather_write(struct device *dev,
- const void *reg, size_t reg_size,
- const void *val, size_t val_size)
-{
- struct i2c_client *i2c = to_i2c_client(dev);
- struct i2c_msg xfer[2];
- int ret;
-
- /* If the I2C controller can't do a gather tell the core, it
- * will substitute in a linear write for us.
- */
- if (!i2c_check_functionality(i2c->adapter, I2C_FUNC_PROTOCOL_MANGLING))
- return -ENOTSUPP;
-
- xfer[0].addr = i2c->addr;
- xfer[0].flags = 0;
- xfer[0].len = reg_size;
- xfer[0].buf = (void *)reg;
-
- xfer[1].addr = i2c->addr;
- xfer[1].flags = I2C_M_NOSTART;
- xfer[1].len = val_size;
- xfer[1].buf = (void *)val;
-
- ret = i2c_transfer(i2c->adapter, xfer, 2);
- if (ret == 2)
- return 0;
- if (ret < 0)
- return ret;
- else
- return -EIO;
-}
-
-static int regmap_i2c_read(struct device *dev,
- const void *reg, size_t reg_size,
- void *val, size_t val_size)
-{
- struct i2c_client *i2c = to_i2c_client(dev);
- struct i2c_msg xfer[2];
- int ret;
-
- xfer[0].addr = i2c->addr;
- xfer[0].flags = 0;
- xfer[0].len = reg_size;
- xfer[0].buf = (void *)reg;
-
- xfer[1].addr = i2c->addr;
- xfer[1].flags = I2C_M_RD;
- xfer[1].len = val_size;
- xfer[1].buf = val;
-
- ret = i2c_transfer(i2c->adapter, xfer, 2);
- if (ret == 2)
- return 0;
- else if (ret < 0)
- return ret;
- else
- return -EIO;
-}
-
-static struct regmap_bus regmap_i2c = {
- .write = regmap_i2c_write,
- .gather_write = regmap_i2c_gather_write,
- .read = regmap_i2c_read,
-};
-
-/**
- * regmap_init_i2c(): Initialise register map
- *
- * @i2c: Device that will be interacted with
- * @config: Configuration for register map
- *
- * The return value will be an ERR_PTR() on error or a valid pointer to
- * a struct regmap.
- */
-struct regmap *regmap_init_i2c(struct i2c_client *i2c,
- const struct regmap_config *config)
-{
- return regmap_init(&i2c->dev, &regmap_i2c, config);
-}
-EXPORT_SYMBOL_GPL(regmap_init_i2c);
-
-/**
- * devm_regmap_init_i2c(): Initialise managed register map
- *
- * @i2c: Device that will be interacted with
- * @config: Configuration for register map
- *
- * The return value will be an ERR_PTR() on error or a valid pointer
- * to a struct regmap. The regmap will be automatically freed by the
- * device management code.
- */
-struct regmap *devm_regmap_init_i2c(struct i2c_client *i2c,
- const struct regmap_config *config)
-{
- return devm_regmap_init(&i2c->dev, &regmap_i2c, config);
-}
-EXPORT_SYMBOL_GPL(devm_regmap_init_i2c);
-
-MODULE_LICENSE("GPL");
diff --git a/ANDROID_3.4.5/drivers/base/regmap/regmap-irq.c b/ANDROID_3.4.5/drivers/base/regmap/regmap-irq.c
deleted file mode 100644
index 1befaa7a..00000000
--- a/ANDROID_3.4.5/drivers/base/regmap/regmap-irq.c
+++ /dev/null
@@ -1,303 +0,0 @@
-/*
- * regmap based irq_chip
- *
- * Copyright 2011 Wolfson Microelectronics plc
- *
- * Author: Mark Brown <broonie@opensource.wolfsonmicro.com>
- *
- * This program is free software; you can redistribute it and/or modify
- * it under the terms of the GNU General Public License version 2 as
- * published by the Free Software Foundation.
- */
-
-#include <linux/export.h>
-#include <linux/device.h>
-#include <linux/regmap.h>
-#include <linux/irq.h>
-#include <linux/interrupt.h>
-#include <linux/slab.h>
-
-#include "internal.h"
-
-struct regmap_irq_chip_data {
- struct mutex lock;
-
- struct regmap *map;
- struct regmap_irq_chip *chip;
-
- int irq_base;
-
- void *status_reg_buf;
- unsigned int *status_buf;
- unsigned int *mask_buf;
- unsigned int *mask_buf_def;
-};
-
-static inline const
-struct regmap_irq *irq_to_regmap_irq(struct regmap_irq_chip_data *data,
- int irq)
-{
- return &data->chip->irqs[irq - data->irq_base];
-}
-
-static void regmap_irq_lock(struct irq_data *data)
-{
- struct regmap_irq_chip_data *d = irq_data_get_irq_chip_data(data);
-
- mutex_lock(&d->lock);
-}
-
-static void regmap_irq_sync_unlock(struct irq_data *data)
-{
- struct regmap_irq_chip_data *d = irq_data_get_irq_chip_data(data);
- int i, ret;
-
- /*
- * If there's been a change in the mask write it back to the
- * hardware. We rely on the use of the regmap core cache to
- * suppress pointless writes.
- */
- for (i = 0; i < d->chip->num_regs; i++) {
- ret = regmap_update_bits(d->map, d->chip->mask_base + i,
- d->mask_buf_def[i], d->mask_buf[i]);
- if (ret != 0)
- dev_err(d->map->dev, "Failed to sync masks in %x\n",
- d->chip->mask_base + i);
- }
-
- mutex_unlock(&d->lock);
-}
-
-static void regmap_irq_enable(struct irq_data *data)
-{
- struct regmap_irq_chip_data *d = irq_data_get_irq_chip_data(data);
- const struct regmap_irq *irq_data = irq_to_regmap_irq(d, data->irq);
-
- d->mask_buf[irq_data->reg_offset] &= ~irq_data->mask;
-}
-
-static void regmap_irq_disable(struct irq_data *data)
-{
- struct regmap_irq_chip_data *d = irq_data_get_irq_chip_data(data);
- const struct regmap_irq *irq_data = irq_to_regmap_irq(d, data->irq);
-
- d->mask_buf[irq_data->reg_offset] |= irq_data->mask;
-}
-
-static struct irq_chip regmap_irq_chip = {
- .name = "regmap",
- .irq_bus_lock = regmap_irq_lock,
- .irq_bus_sync_unlock = regmap_irq_sync_unlock,
- .irq_disable = regmap_irq_disable,
- .irq_enable = regmap_irq_enable,
-};
-
-static irqreturn_t regmap_irq_thread(int irq, void *d)
-{
- struct regmap_irq_chip_data *data = d;
- struct regmap_irq_chip *chip = data->chip;
- struct regmap *map = data->map;
- int ret, i;
- u8 *buf8 = data->status_reg_buf;
- u16 *buf16 = data->status_reg_buf;
- u32 *buf32 = data->status_reg_buf;
- bool handled = false;
-
- ret = regmap_bulk_read(map, chip->status_base, data->status_reg_buf,
- chip->num_regs);
- if (ret != 0) {
- dev_err(map->dev, "Failed to read IRQ status: %d\n", ret);
- return IRQ_NONE;
- }
-
- /*
- * Ignore masked IRQs and ack if we need to; we ack early so
- * there is no race between handling and acknowleding the
- * interrupt. We assume that typically few of the interrupts
- * will fire simultaneously so don't worry about overhead from
- * doing a write per register.
- */
- for (i = 0; i < data->chip->num_regs; i++) {
- switch (map->format.val_bytes) {
- case 1:
- data->status_buf[i] = buf8[i];
- break;
- case 2:
- data->status_buf[i] = buf16[i];
- break;
- case 4:
- data->status_buf[i] = buf32[i];
- break;
- default:
- BUG();
- return IRQ_NONE;
- }
-
- data->status_buf[i] &= ~data->mask_buf[i];
-
- if (data->status_buf[i] && chip->ack_base) {
- ret = regmap_write(map, chip->ack_base + i,
- data->status_buf[i]);
- if (ret != 0)
- dev_err(map->dev, "Failed to ack 0x%x: %d\n",
- chip->ack_base + i, ret);
- }
- }
-
- for (i = 0; i < chip->num_irqs; i++) {
- if (data->status_buf[chip->irqs[i].reg_offset] &
- chip->irqs[i].mask) {
- handle_nested_irq(data->irq_base + i);
- handled = true;
- }
- }
-
- if (handled)
- return IRQ_HANDLED;
- else
- return IRQ_NONE;
-}
-
-/**
- * regmap_add_irq_chip(): Use standard regmap IRQ controller handling
- *
- * map: The regmap for the device.
- * irq: The IRQ the device uses to signal interrupts
- * irq_flags: The IRQF_ flags to use for the primary interrupt.
- * chip: Configuration for the interrupt controller.
- * data: Runtime data structure for the controller, allocated on success
- *
- * Returns 0 on success or an errno on failure.
- *
- * In order for this to be efficient the chip really should use a
- * register cache. The chip driver is responsible for restoring the
- * register values used by the IRQ controller over suspend and resume.
- */
-int regmap_add_irq_chip(struct regmap *map, int irq, int irq_flags,
- int irq_base, struct regmap_irq_chip *chip,
- struct regmap_irq_chip_data **data)
-{
- struct regmap_irq_chip_data *d;
- int cur_irq, i;
- int ret = -ENOMEM;
-
- irq_base = irq_alloc_descs(irq_base, 0, chip->num_irqs, 0);
- if (irq_base < 0) {
- dev_warn(map->dev, "Failed to allocate IRQs: %d\n",
- irq_base);
- return irq_base;
- }
-
- d = kzalloc(sizeof(*d), GFP_KERNEL);
- if (!d)
- return -ENOMEM;
-
- d->status_buf = kzalloc(sizeof(unsigned int) * chip->num_regs,
- GFP_KERNEL);
- if (!d->status_buf)
- goto err_alloc;
-
- d->status_reg_buf = kzalloc(map->format.val_bytes * chip->num_regs,
- GFP_KERNEL);
- if (!d->status_reg_buf)
- goto err_alloc;
-
- d->mask_buf = kzalloc(sizeof(unsigned int) * chip->num_regs,
- GFP_KERNEL);
- if (!d->mask_buf)
- goto err_alloc;
-
- d->mask_buf_def = kzalloc(sizeof(unsigned int) * chip->num_regs,
- GFP_KERNEL);
- if (!d->mask_buf_def)
- goto err_alloc;
-
- d->map = map;
- d->chip = chip;
- d->irq_base = irq_base;
- mutex_init(&d->lock);
-
- for (i = 0; i < chip->num_irqs; i++)
- d->mask_buf_def[chip->irqs[i].reg_offset]
- |= chip->irqs[i].mask;
-
- /* Mask all the interrupts by default */
- for (i = 0; i < chip->num_regs; i++) {
- d->mask_buf[i] = d->mask_buf_def[i];
- ret = regmap_write(map, chip->mask_base + i, d->mask_buf[i]);
- if (ret != 0) {
- dev_err(map->dev, "Failed to set masks in 0x%x: %d\n",
- chip->mask_base + i, ret);
- goto err_alloc;
- }
- }
-
- /* Register them with genirq */
- for (cur_irq = irq_base;
- cur_irq < chip->num_irqs + irq_base;
- cur_irq++) {
- irq_set_chip_data(cur_irq, d);
- irq_set_chip_and_handler(cur_irq, &regmap_irq_chip,
- handle_edge_irq);
- irq_set_nested_thread(cur_irq, 1);
-
- /* ARM needs us to explicitly flag the IRQ as valid
- * and will set them noprobe when we do so. */
-#ifdef CONFIG_ARM
- set_irq_flags(cur_irq, IRQF_VALID);
-#else
- irq_set_noprobe(cur_irq);
-#endif
- }
-
- ret = request_threaded_irq(irq, NULL, regmap_irq_thread, irq_flags,
- chip->name, d);
- if (ret != 0) {
- dev_err(map->dev, "Failed to request IRQ %d: %d\n", irq, ret);
- goto err_alloc;
- }
-
- return 0;
-
-err_alloc:
- kfree(d->mask_buf_def);
- kfree(d->mask_buf);
- kfree(d->status_reg_buf);
- kfree(d->status_buf);
- kfree(d);
- return ret;
-}
-EXPORT_SYMBOL_GPL(regmap_add_irq_chip);
-
-/**
- * regmap_del_irq_chip(): Stop interrupt handling for a regmap IRQ chip
- *
- * @irq: Primary IRQ for the device
- * @d: regmap_irq_chip_data allocated by regmap_add_irq_chip()
- */
-void regmap_del_irq_chip(int irq, struct regmap_irq_chip_data *d)
-{
- if (!d)
- return;
-
- free_irq(irq, d);
- kfree(d->mask_buf_def);
- kfree(d->mask_buf);
- kfree(d->status_reg_buf);
- kfree(d->status_buf);
- kfree(d);
-}
-EXPORT_SYMBOL_GPL(regmap_del_irq_chip);
-
-/**
- * regmap_irq_chip_get_base(): Retrieve interrupt base for a regmap IRQ chip
- *
- * Useful for drivers to request their own IRQs.
- *
- * @data: regmap_irq controller to operate on.
- */
-int regmap_irq_chip_get_base(struct regmap_irq_chip_data *data)
-{
- return data->irq_base;
-}
-EXPORT_SYMBOL_GPL(regmap_irq_chip_get_base);
diff --git a/ANDROID_3.4.5/drivers/base/regmap/regmap-spi.c b/ANDROID_3.4.5/drivers/base/regmap/regmap-spi.c
deleted file mode 100644
index 7c0c35a3..00000000
--- a/ANDROID_3.4.5/drivers/base/regmap/regmap-spi.c
+++ /dev/null
@@ -1,90 +0,0 @@
-/*
- * Register map access API - SPI support
- *
- * Copyright 2011 Wolfson Microelectronics plc
- *
- * Author: Mark Brown <broonie@opensource.wolfsonmicro.com>
- *
- * This program is free software; you can redistribute it and/or modify
- * it under the terms of the GNU General Public License version 2 as
- * published by the Free Software Foundation.
- */
-
-#include <linux/regmap.h>
-#include <linux/spi/spi.h>
-#include <linux/init.h>
-#include <linux/module.h>
-
-static int regmap_spi_write(struct device *dev, const void *data, size_t count)
-{
- struct spi_device *spi = to_spi_device(dev);
-
- return spi_write(spi, data, count);
-}
-
-static int regmap_spi_gather_write(struct device *dev,
- const void *reg, size_t reg_len,
- const void *val, size_t val_len)
-{
- struct spi_device *spi = to_spi_device(dev);
- struct spi_message m;
- struct spi_transfer t[2] = { { .tx_buf = reg, .len = reg_len, },
- { .tx_buf = val, .len = val_len, }, };
-
- spi_message_init(&m);
- spi_message_add_tail(&t[0], &m);
- spi_message_add_tail(&t[1], &m);
-
- return spi_sync(spi, &m);
-}
-
-static int regmap_spi_read(struct device *dev,
- const void *reg, size_t reg_size,
- void *val, size_t val_size)
-{
- struct spi_device *spi = to_spi_device(dev);
-
- return spi_write_then_read(spi, reg, reg_size, val, val_size);
-}
-
-static struct regmap_bus regmap_spi = {
- .write = regmap_spi_write,
- .gather_write = regmap_spi_gather_write,
- .read = regmap_spi_read,
- .read_flag_mask = 0x80,
-};
-
-/**
- * regmap_init_spi(): Initialise register map
- *
- * @spi: Device that will be interacted with
- * @config: Configuration for register map
- *
- * The return value will be an ERR_PTR() on error or a valid pointer to
- * a struct regmap.
- */
-struct regmap *regmap_init_spi(struct spi_device *spi,
- const struct regmap_config *config)
-{
- return regmap_init(&spi->dev, &regmap_spi, config);
-}
-EXPORT_SYMBOL_GPL(regmap_init_spi);
-
-/**
- * devm_regmap_init_spi(): Initialise register map
- *
- * @spi: Device that will be interacted with
- * @config: Configuration for register map
- *
- * The return value will be an ERR_PTR() on error or a valid pointer
- * to a struct regmap. The map will be automatically freed by the
- * device management code.
- */
-struct regmap *devm_regmap_init_spi(struct spi_device *spi,
- const struct regmap_config *config)
-{
- return devm_regmap_init(&spi->dev, &regmap_spi, config);
-}
-EXPORT_SYMBOL_GPL(devm_regmap_init_spi);
-
-MODULE_LICENSE("GPL");
diff --git a/ANDROID_3.4.5/drivers/base/regmap/regmap.c b/ANDROID_3.4.5/drivers/base/regmap/regmap.c
deleted file mode 100644
index bb80853f..00000000
--- a/ANDROID_3.4.5/drivers/base/regmap/regmap.c
+++ /dev/null
@@ -1,936 +0,0 @@
-/*
- * Register map access API
- *
- * Copyright 2011 Wolfson Microelectronics plc
- *
- * Author: Mark Brown <broonie@opensource.wolfsonmicro.com>
- *
- * This program is free software; you can redistribute it and/or modify
- * it under the terms of the GNU General Public License version 2 as
- * published by the Free Software Foundation.
- */
-
-#include <linux/device.h>
-#include <linux/slab.h>
-#include <linux/export.h>
-#include <linux/mutex.h>
-#include <linux/err.h>
-
-#define CREATE_TRACE_POINTS
-#include <trace/events/regmap.h>
-
-#include "internal.h"
-
-bool regmap_writeable(struct regmap *map, unsigned int reg)
-{
- if (map->max_register && reg > map->max_register)
- return false;
-
- if (map->writeable_reg)
- return map->writeable_reg(map->dev, reg);
-
- return true;
-}
-
-bool regmap_readable(struct regmap *map, unsigned int reg)
-{
- if (map->max_register && reg > map->max_register)
- return false;
-
- if (map->format.format_write)
- return false;
-
- if (map->readable_reg)
- return map->readable_reg(map->dev, reg);
-
- return true;
-}
-
-bool regmap_volatile(struct regmap *map, unsigned int reg)
-{
- if (!regmap_readable(map, reg))
- return false;
-
- if (map->volatile_reg)
- return map->volatile_reg(map->dev, reg);
-
- return true;
-}
-
-bool regmap_precious(struct regmap *map, unsigned int reg)
-{
- if (!regmap_readable(map, reg))
- return false;
-
- if (map->precious_reg)
- return map->precious_reg(map->dev, reg);
-
- return false;
-}
-
-static bool regmap_volatile_range(struct regmap *map, unsigned int reg,
- unsigned int num)
-{
- unsigned int i;
-
- for (i = 0; i < num; i++)
- if (!regmap_volatile(map, reg + i))
- return false;
-
- return true;
-}
-
-static void regmap_format_2_6_write(struct regmap *map,
- unsigned int reg, unsigned int val)
-{
- u8 *out = map->work_buf;
-
- *out = (reg << 6) | val;
-}
-
-static void regmap_format_4_12_write(struct regmap *map,
- unsigned int reg, unsigned int val)
-{
- __be16 *out = map->work_buf;
- *out = cpu_to_be16((reg << 12) | val);
-}
-
-static void regmap_format_7_9_write(struct regmap *map,
- unsigned int reg, unsigned int val)
-{
- __be16 *out = map->work_buf;
- *out = cpu_to_be16((reg << 9) | val);
-}
-
-static void regmap_format_10_14_write(struct regmap *map,
- unsigned int reg, unsigned int val)
-{
- u8 *out = map->work_buf;
-
- out[2] = val;
- out[1] = (val >> 8) | (reg << 6);
- out[0] = reg >> 2;
-}
-
-static void regmap_format_8(void *buf, unsigned int val)
-{
- u8 *b = buf;
-
- b[0] = val;
-}
-
-static void regmap_format_16(void *buf, unsigned int val)
-{
- __be16 *b = buf;
-
- b[0] = cpu_to_be16(val);
-}
-
-static void regmap_format_32(void *buf, unsigned int val)
-{
- __be32 *b = buf;
-
- b[0] = cpu_to_be32(val);
-}
-
-static unsigned int regmap_parse_8(void *buf)
-{
- u8 *b = buf;
-
- return b[0];
-}
-
-static unsigned int regmap_parse_16(void *buf)
-{
- __be16 *b = buf;
-
- b[0] = be16_to_cpu(b[0]);
-
- return b[0];
-}
-
-static unsigned int regmap_parse_32(void *buf)
-{
- __be32 *b = buf;
-
- b[0] = be32_to_cpu(b[0]);
-
- return b[0];
-}
-
-/**
- * regmap_init(): Initialise register map
- *
- * @dev: Device that will be interacted with
- * @bus: Bus-specific callbacks to use with device
- * @config: Configuration for register map
- *
- * The return value will be an ERR_PTR() on error or a valid pointer to
- * a struct regmap. This function should generally not be called
- * directly, it should be called by bus-specific init functions.
- */
-struct regmap *regmap_init(struct device *dev,
- const struct regmap_bus *bus,
- const struct regmap_config *config)
-{
- struct regmap *map;
- int ret = -EINVAL;
-
- if (!bus || !config)
- goto err;
-
- map = kzalloc(sizeof(*map), GFP_KERNEL);
- if (map == NULL) {
- ret = -ENOMEM;
- goto err;
- }
-
- mutex_init(&map->lock);
- map->format.buf_size = (config->reg_bits + config->val_bits) / 8;
- map->format.reg_bytes = DIV_ROUND_UP(config->reg_bits, 8);
- map->format.pad_bytes = config->pad_bits / 8;
- map->format.val_bytes = DIV_ROUND_UP(config->val_bits, 8);
- map->format.buf_size += map->format.pad_bytes;
- map->dev = dev;
- map->bus = bus;
- map->max_register = config->max_register;
- map->writeable_reg = config->writeable_reg;
- map->readable_reg = config->readable_reg;
- map->volatile_reg = config->volatile_reg;
- map->precious_reg = config->precious_reg;
- map->cache_type = config->cache_type;
-
- if (config->read_flag_mask || config->write_flag_mask) {
- map->read_flag_mask = config->read_flag_mask;
- map->write_flag_mask = config->write_flag_mask;
- } else {
- map->read_flag_mask = bus->read_flag_mask;
- }
-
- switch (config->reg_bits) {
- case 2:
- switch (config->val_bits) {
- case 6:
- map->format.format_write = regmap_format_2_6_write;
- break;
- default:
- goto err_map;
- }
- break;
-
- case 4:
- switch (config->val_bits) {
- case 12:
- map->format.format_write = regmap_format_4_12_write;
- break;
- default:
- goto err_map;
- }
- break;
-
- case 7:
- switch (config->val_bits) {
- case 9:
- map->format.format_write = regmap_format_7_9_write;
- break;
- default:
- goto err_map;
- }
- break;
-
- case 10:
- switch (config->val_bits) {
- case 14:
- map->format.format_write = regmap_format_10_14_write;
- break;
- default:
- goto err_map;
- }
- break;
-
- case 8:
- map->format.format_reg = regmap_format_8;
- break;
-
- case 16:
- map->format.format_reg = regmap_format_16;
- break;
-
- case 32:
- map->format.format_reg = regmap_format_32;
- break;
-
- default:
- goto err_map;
- }
-
- switch (config->val_bits) {
- case 8:
- map->format.format_val = regmap_format_8;
- map->format.parse_val = regmap_parse_8;
- break;
- case 16:
- map->format.format_val = regmap_format_16;
- map->format.parse_val = regmap_parse_16;
- break;
- case 32:
- map->format.format_val = regmap_format_32;
- map->format.parse_val = regmap_parse_32;
- break;
- }
-
- if (!map->format.format_write &&
- !(map->format.format_reg && map->format.format_val))
- goto err_map;
-
- map->work_buf = kzalloc(map->format.buf_size, GFP_KERNEL);
- if (map->work_buf == NULL) {
- ret = -ENOMEM;
- goto err_map;
- }
-
- regmap_debugfs_init(map);
-
- ret = regcache_init(map, config);
- if (ret < 0)
- goto err_free_workbuf;
-
- return map;
-
-err_free_workbuf:
- kfree(map->work_buf);
-err_map:
- kfree(map);
-err:
- return ERR_PTR(ret);
-}
-EXPORT_SYMBOL_GPL(regmap_init);
-
-static void devm_regmap_release(struct device *dev, void *res)
-{
- regmap_exit(*(struct regmap **)res);
-}
-
-/**
- * devm_regmap_init(): Initialise managed register map
- *
- * @dev: Device that will be interacted with
- * @bus: Bus-specific callbacks to use with device
- * @config: Configuration for register map
- *
- * The return value will be an ERR_PTR() on error or a valid pointer
- * to a struct regmap. This function should generally not be called
- * directly, it should be called by bus-specific init functions. The
- * map will be automatically freed by the device management code.
- */
-struct regmap *devm_regmap_init(struct device *dev,
- const struct regmap_bus *bus,
- const struct regmap_config *config)
-{
- struct regmap **ptr, *regmap;
-
- ptr = devres_alloc(devm_regmap_release, sizeof(*ptr), GFP_KERNEL);
- if (!ptr)
- return ERR_PTR(-ENOMEM);
-
- regmap = regmap_init(dev, bus, config);
- if (!IS_ERR(regmap)) {
- *ptr = regmap;
- devres_add(dev, ptr);
- } else {
- devres_free(ptr);
- }
-
- return regmap;
-}
-EXPORT_SYMBOL_GPL(devm_regmap_init);
-
-/**
- * regmap_reinit_cache(): Reinitialise the current register cache
- *
- * @map: Register map to operate on.
- * @config: New configuration. Only the cache data will be used.
- *
- * Discard any existing register cache for the map and initialize a
- * new cache. This can be used to restore the cache to defaults or to
- * update the cache configuration to reflect runtime discovery of the
- * hardware.
- */
-int regmap_reinit_cache(struct regmap *map, const struct regmap_config *config)
-{
- int ret;
-
- mutex_lock(&map->lock);
-
- regcache_exit(map);
- regmap_debugfs_exit(map);
-
- map->max_register = config->max_register;
- map->writeable_reg = config->writeable_reg;
- map->readable_reg = config->readable_reg;
- map->volatile_reg = config->volatile_reg;
- map->precious_reg = config->precious_reg;
- map->cache_type = config->cache_type;
-
- regmap_debugfs_init(map);
-
- map->cache_bypass = false;
- map->cache_only = false;
-
- ret = regcache_init(map, config);
-
- mutex_unlock(&map->lock);
-
- return ret;
-}
-
-/**
- * regmap_exit(): Free a previously allocated register map
- */
-void regmap_exit(struct regmap *map)
-{
- regcache_exit(map);
- regmap_debugfs_exit(map);
- kfree(map->work_buf);
- kfree(map);
-}
-EXPORT_SYMBOL_GPL(regmap_exit);
-
-static int _regmap_raw_write(struct regmap *map, unsigned int reg,
- const void *val, size_t val_len)
-{
- u8 *u8 = map->work_buf;
- void *buf;
- int ret = -ENOTSUPP;
- size_t len;
- int i;
-
- /* Check for unwritable registers before we start */
- if (map->writeable_reg)
- for (i = 0; i < val_len / map->format.val_bytes; i++)
- if (!map->writeable_reg(map->dev, reg + i))
- return -EINVAL;
-
- if (!map->cache_bypass && map->format.parse_val) {
- unsigned int ival;
- int val_bytes = map->format.val_bytes;
- for (i = 0; i < val_len / val_bytes; i++) {
- memcpy(map->work_buf, val + (i * val_bytes), val_bytes);
- ival = map->format.parse_val(map->work_buf);
- ret = regcache_write(map, reg + i, ival);
- if (ret) {
- dev_err(map->dev,
- "Error in caching of register: %u ret: %d\n",
- reg + i, ret);
- return ret;
- }
- }
- if (map->cache_only) {
- map->cache_dirty = true;
- return 0;
- }
- }
-
- map->format.format_reg(map->work_buf, reg);
-
- u8[0] |= map->write_flag_mask;
-
- trace_regmap_hw_write_start(map->dev, reg,
- val_len / map->format.val_bytes);
-
- /* If we're doing a single register write we can probably just
- * send the work_buf directly, otherwise try to do a gather
- * write.
- */
- if (val == (map->work_buf + map->format.pad_bytes +
- map->format.reg_bytes))
- ret = map->bus->write(map->dev, map->work_buf,
- map->format.reg_bytes +
- map->format.pad_bytes +
- val_len);
- else if (map->bus->gather_write)
- ret = map->bus->gather_write(map->dev, map->work_buf,
- map->format.reg_bytes +
- map->format.pad_bytes,
- val, val_len);
-
- /* If that didn't work fall back on linearising by hand. */
- if (ret == -ENOTSUPP) {
- len = map->format.reg_bytes + map->format.pad_bytes + val_len;
- buf = kzalloc(len, GFP_KERNEL);
- if (!buf)
- return -ENOMEM;
-
- memcpy(buf, map->work_buf, map->format.reg_bytes);
- memcpy(buf + map->format.reg_bytes + map->format.pad_bytes,
- val, val_len);
- ret = map->bus->write(map->dev, buf, len);
-
- kfree(buf);
- }
-
- trace_regmap_hw_write_done(map->dev, reg,
- val_len / map->format.val_bytes);
-
- return ret;
-}
-
-int _regmap_write(struct regmap *map, unsigned int reg,
- unsigned int val)
-{
- int ret;
- BUG_ON(!map->format.format_write && !map->format.format_val);
-
- if (!map->cache_bypass && map->format.format_write) {
- ret = regcache_write(map, reg, val);
- if (ret != 0)
- return ret;
- if (map->cache_only) {
- map->cache_dirty = true;
- return 0;
- }
- }
-
- trace_regmap_reg_write(map->dev, reg, val);
-
- if (map->format.format_write) {
- map->format.format_write(map, reg, val);
-
- trace_regmap_hw_write_start(map->dev, reg, 1);
-
- ret = map->bus->write(map->dev, map->work_buf,
- map->format.buf_size);
-
- trace_regmap_hw_write_done(map->dev, reg, 1);
-
- return ret;
- } else {
- map->format.format_val(map->work_buf + map->format.reg_bytes
- + map->format.pad_bytes, val);
- return _regmap_raw_write(map, reg,
- map->work_buf +
- map->format.reg_bytes +
- map->format.pad_bytes,
- map->format.val_bytes);
- }
-}
-
-/**
- * regmap_write(): Write a value to a single register
- *
- * @map: Register map to write to
- * @reg: Register to write to
- * @val: Value to be written
- *
- * A value of zero will be returned on success, a negative errno will
- * be returned in error cases.
- */
-int regmap_write(struct regmap *map, unsigned int reg, unsigned int val)
-{
- int ret;
-
- mutex_lock(&map->lock);
-
- ret = _regmap_write(map, reg, val);
-
- mutex_unlock(&map->lock);
-
- return ret;
-}
-EXPORT_SYMBOL_GPL(regmap_write);
-
-/**
- * regmap_raw_write(): Write raw values to one or more registers
- *
- * @map: Register map to write to
- * @reg: Initial register to write to
- * @val: Block of data to be written, laid out for direct transmission to the
- * device
- * @val_len: Length of data pointed to by val.
- *
- * This function is intended to be used for things like firmware
- * download where a large block of data needs to be transferred to the
- * device. No formatting will be done on the data provided.
- *
- * A value of zero will be returned on success, a negative errno will
- * be returned in error cases.
- */
-int regmap_raw_write(struct regmap *map, unsigned int reg,
- const void *val, size_t val_len)
-{
- int ret;
-
- mutex_lock(&map->lock);
-
- ret = _regmap_raw_write(map, reg, val, val_len);
-
- mutex_unlock(&map->lock);
-
- return ret;
-}
-EXPORT_SYMBOL_GPL(regmap_raw_write);
-
-/*
- * regmap_bulk_write(): Write multiple registers to the device
- *
- * @map: Register map to write to
- * @reg: First register to be write from
- * @val: Block of data to be written, in native register size for device
- * @val_count: Number of registers to write
- *
- * This function is intended to be used for writing a large block of
- * data to be device either in single transfer or multiple transfer.
- *
- * A value of zero will be returned on success, a negative errno will
- * be returned in error cases.
- */
-int regmap_bulk_write(struct regmap *map, unsigned int reg, const void *val,
- size_t val_count)
-{
- int ret = 0, i;
- size_t val_bytes = map->format.val_bytes;
- void *wval;
-
- if (!map->format.parse_val)
- return -EINVAL;
-
- mutex_lock(&map->lock);
-
- /* No formatting is require if val_byte is 1 */
- if (val_bytes == 1) {
- wval = (void *)val;
- } else {
- wval = kmemdup(val, val_count * val_bytes, GFP_KERNEL);
- if (!wval) {
- ret = -ENOMEM;
- dev_err(map->dev, "Error in memory allocation\n");
- goto out;
- }
- for (i = 0; i < val_count * val_bytes; i += val_bytes)
- map->format.parse_val(wval + i);
- }
- ret = _regmap_raw_write(map, reg, wval, val_bytes * val_count);
-
- if (val_bytes != 1)
- kfree(wval);
-
-out:
- mutex_unlock(&map->lock);
- return ret;
-}
-EXPORT_SYMBOL_GPL(regmap_bulk_write);
-
-static int _regmap_raw_read(struct regmap *map, unsigned int reg, void *val,
- unsigned int val_len)
-{
- u8 *u8 = map->work_buf;
- int ret;
-
- map->format.format_reg(map->work_buf, reg);
-
- /*
- * Some buses or devices flag reads by setting the high bits in the
- * register addresss; since it's always the high bits for all
- * current formats we can do this here rather than in
- * formatting. This may break if we get interesting formats.
- */
- u8[0] |= map->read_flag_mask;
-
- trace_regmap_hw_read_start(map->dev, reg,
- val_len / map->format.val_bytes);
-
- ret = map->bus->read(map->dev, map->work_buf,
- map->format.reg_bytes + map->format.pad_bytes,
- val, val_len);
-
- trace_regmap_hw_read_done(map->dev, reg,
- val_len / map->format.val_bytes);
-
- return ret;
-}
-
-static int _regmap_read(struct regmap *map, unsigned int reg,
- unsigned int *val)
-{
- int ret;
-
- if (!map->cache_bypass) {
- ret = regcache_read(map, reg, val);
- if (ret == 0)
- return 0;
- }
-
- if (!map->format.parse_val)
- return -EINVAL;
-
- if (map->cache_only)
- return -EBUSY;
-
- ret = _regmap_raw_read(map, reg, map->work_buf, map->format.val_bytes);
- if (ret == 0) {
- *val = map->format.parse_val(map->work_buf);
- trace_regmap_reg_read(map->dev, reg, *val);
- }
-
- return ret;
-}
-
-/**
- * regmap_read(): Read a value from a single register
- *
- * @map: Register map to write to
- * @reg: Register to be read from
- * @val: Pointer to store read value
- *
- * A value of zero will be returned on success, a negative errno will
- * be returned in error cases.
- */
-int regmap_read(struct regmap *map, unsigned int reg, unsigned int *val)
-{
- int ret;
-
- mutex_lock(&map->lock);
-
- ret = _regmap_read(map, reg, val);
-
- mutex_unlock(&map->lock);
-
- return ret;
-}
-EXPORT_SYMBOL_GPL(regmap_read);
-
-/**
- * regmap_raw_read(): Read raw data from the device
- *
- * @map: Register map to write to
- * @reg: First register to be read from
- * @val: Pointer to store read value
- * @val_len: Size of data to read
- *
- * A value of zero will be returned on success, a negative errno will
- * be returned in error cases.
- */
-int regmap_raw_read(struct regmap *map, unsigned int reg, void *val,
- size_t val_len)
-{
- size_t val_bytes = map->format.val_bytes;
- size_t val_count = val_len / val_bytes;
- unsigned int v;
- int ret, i;
-
- mutex_lock(&map->lock);
-
- if (regmap_volatile_range(map, reg, val_count) || map->cache_bypass ||
- map->cache_type == REGCACHE_NONE) {
- /* Physical block read if there's no cache involved */
- ret = _regmap_raw_read(map, reg, val, val_len);
-
- } else {
- /* Otherwise go word by word for the cache; should be low
- * cost as we expect to hit the cache.
- */
- for (i = 0; i < val_count; i++) {
- ret = _regmap_read(map, reg + i, &v);
- if (ret != 0)
- goto out;
-
- map->format.format_val(val + (i * val_bytes), v);
- }
- }
-
- out:
- mutex_unlock(&map->lock);
-
- return ret;
-}
-EXPORT_SYMBOL_GPL(regmap_raw_read);
-
-/**
- * regmap_bulk_read(): Read multiple registers from the device
- *
- * @map: Register map to write to
- * @reg: First register to be read from
- * @val: Pointer to store read value, in native register size for device
- * @val_count: Number of registers to read
- *
- * A value of zero will be returned on success, a negative errno will
- * be returned in error cases.
- */
-int regmap_bulk_read(struct regmap *map, unsigned int reg, void *val,
- size_t val_count)
-{
- int ret, i;
- size_t val_bytes = map->format.val_bytes;
- bool vol = regmap_volatile_range(map, reg, val_count);
-
- if (!map->format.parse_val)
- return -EINVAL;
-
- if (vol || map->cache_type == REGCACHE_NONE) {
- ret = regmap_raw_read(map, reg, val, val_bytes * val_count);
- if (ret != 0)
- return ret;
-
- for (i = 0; i < val_count * val_bytes; i += val_bytes)
- map->format.parse_val(val + i);
- } else {
- for (i = 0; i < val_count; i++) {
- unsigned int ival;
- ret = regmap_read(map, reg + i, &ival);
- if (ret != 0)
- return ret;
- memcpy(val + (i * val_bytes), &ival, val_bytes);
- }
- }
-
- return 0;
-}
-EXPORT_SYMBOL_GPL(regmap_bulk_read);
-
-static int _regmap_update_bits(struct regmap *map, unsigned int reg,
- unsigned int mask, unsigned int val,
- bool *change)
-{
- int ret;
- unsigned int tmp, orig;
-
- mutex_lock(&map->lock);
-
- ret = _regmap_read(map, reg, &orig);
- if (ret != 0)
- goto out;
-
- tmp = orig & ~mask;
- tmp |= val & mask;
-
- if (tmp != orig) {
- ret = _regmap_write(map, reg, tmp);
- *change = true;
- } else {
- *change = false;
- }
-
-out:
- mutex_unlock(&map->lock);
-
- return ret;
-}
-
-/**
- * regmap_update_bits: Perform a read/modify/write cycle on the register map
- *
- * @map: Register map to update
- * @reg: Register to update
- * @mask: Bitmask to change
- * @val: New value for bitmask
- *
- * Returns zero for success, a negative number on error.
- */
-int regmap_update_bits(struct regmap *map, unsigned int reg,
- unsigned int mask, unsigned int val)
-{
- bool change;
- return _regmap_update_bits(map, reg, mask, val, &change);
-}
-EXPORT_SYMBOL_GPL(regmap_update_bits);
-
-/**
- * regmap_update_bits_check: Perform a read/modify/write cycle on the
- * register map and report if updated
- *
- * @map: Register map to update
- * @reg: Register to update
- * @mask: Bitmask to change
- * @val: New value for bitmask
- * @change: Boolean indicating if a write was done
- *
- * Returns zero for success, a negative number on error.
- */
-int regmap_update_bits_check(struct regmap *map, unsigned int reg,
- unsigned int mask, unsigned int val,
- bool *change)
-{
- return _regmap_update_bits(map, reg, mask, val, change);
-}
-EXPORT_SYMBOL_GPL(regmap_update_bits_check);
-
-/**
- * regmap_register_patch: Register and apply register updates to be applied
- * on device initialistion
- *
- * @map: Register map to apply updates to.
- * @regs: Values to update.
- * @num_regs: Number of entries in regs.
- *
- * Register a set of register updates to be applied to the device
- * whenever the device registers are synchronised with the cache and
- * apply them immediately. Typically this is used to apply
- * corrections to be applied to the device defaults on startup, such
- * as the updates some vendors provide to undocumented registers.
- */
-int regmap_register_patch(struct regmap *map, const struct reg_default *regs,
- int num_regs)
-{
- int i, ret;
- bool bypass;
-
- /* If needed the implementation can be extended to support this */
- if (map->patch)
- return -EBUSY;
-
- mutex_lock(&map->lock);
-
- bypass = map->cache_bypass;
-
- map->cache_bypass = true;
-
- /* Write out first; it's useful to apply even if we fail later. */
- for (i = 0; i < num_regs; i++) {
- ret = _regmap_write(map, regs[i].reg, regs[i].def);
- if (ret != 0) {
- dev_err(map->dev, "Failed to write %x = %x: %d\n",
- regs[i].reg, regs[i].def, ret);
- goto out;
- }
- }
-
- map->patch = kcalloc(num_regs, sizeof(struct reg_default), GFP_KERNEL);
- if (map->patch != NULL) {
- memcpy(map->patch, regs,
- num_regs * sizeof(struct reg_default));
- map->patch_regs = num_regs;
- } else {
- ret = -ENOMEM;
- }
-
-out:
- map->cache_bypass = bypass;
-
- mutex_unlock(&map->lock);
-
- return ret;
-}
-EXPORT_SYMBOL_GPL(regmap_register_patch);
-
-/*
- * regmap_get_val_bytes(): Report the size of a register value
- *
- * Report the size of a register value, mainly intended to for use by
- * generic infrastructure built on top of regmap.
- */
-int regmap_get_val_bytes(struct regmap *map)
-{
- if (map->format.format_write)
- return -EINVAL;
-
- return map->format.val_bytes;
-}
-EXPORT_SYMBOL_GPL(regmap_get_val_bytes);
-
-static int __init regmap_initcall(void)
-{
- regmap_debugfs_initcall();
-
- return 0;
-}
-postcore_initcall(regmap_initcall);