1
2
3
4
5
6
7
8
9#include <linux/i2c.h>
10#include <linux/spi/spi.h>
11#include <linux/regmap.h>
12#include <linux/export.h>
13#include <sound/soc.h>
14
15
16
17
18
19
20
21
22
23int snd_soc_component_read(struct snd_soc_component *component,
24 unsigned int reg, unsigned int *val)
25{
26 int ret;
27
28 if (component->regmap)
29 ret = regmap_read(component->regmap, reg, val);
30 else if (component->driver->read) {
31 *val = component->driver->read(component, reg);
32 ret = 0;
33 }
34 else
35 ret = -EIO;
36
37 return ret;
38}
39EXPORT_SYMBOL_GPL(snd_soc_component_read);
40
41unsigned int snd_soc_component_read32(struct snd_soc_component *component,
42 unsigned int reg)
43{
44 unsigned int val;
45 int ret;
46
47 ret = snd_soc_component_read(component, reg, &val);
48 if (ret < 0)
49 return -1;
50
51 return val;
52}
53EXPORT_SYMBOL_GPL(snd_soc_component_read32);
54
55
56
57
58
59
60
61
62
63int snd_soc_component_write(struct snd_soc_component *component,
64 unsigned int reg, unsigned int val)
65{
66 if (component->regmap)
67 return regmap_write(component->regmap, reg, val);
68 else if (component->driver->write)
69 return component->driver->write(component, reg, val);
70 else
71 return -EIO;
72}
73EXPORT_SYMBOL_GPL(snd_soc_component_write);
74
75static int snd_soc_component_update_bits_legacy(
76 struct snd_soc_component *component, unsigned int reg,
77 unsigned int mask, unsigned int val, bool *change)
78{
79 unsigned int old, new;
80 int ret;
81
82 mutex_lock(&component->io_mutex);
83
84 ret = snd_soc_component_read(component, reg, &old);
85 if (ret < 0)
86 goto out_unlock;
87
88 new = (old & ~mask) | (val & mask);
89 *change = old != new;
90 if (*change)
91 ret = snd_soc_component_write(component, reg, new);
92out_unlock:
93 mutex_unlock(&component->io_mutex);
94
95 return ret;
96}
97
98
99
100
101
102
103
104
105
106
107
108
109int snd_soc_component_update_bits(struct snd_soc_component *component,
110 unsigned int reg, unsigned int mask, unsigned int val)
111{
112 bool change;
113 int ret;
114
115 if (component->regmap)
116 ret = regmap_update_bits_check(component->regmap, reg, mask,
117 val, &change);
118 else
119 ret = snd_soc_component_update_bits_legacy(component, reg,
120 mask, val, &change);
121
122 if (ret < 0)
123 return ret;
124 return change;
125}
126EXPORT_SYMBOL_GPL(snd_soc_component_update_bits);
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145int snd_soc_component_update_bits_async(struct snd_soc_component *component,
146 unsigned int reg, unsigned int mask, unsigned int val)
147{
148 bool change;
149 int ret;
150
151 if (component->regmap)
152 ret = regmap_update_bits_check_async(component->regmap, reg,
153 mask, val, &change);
154 else
155 ret = snd_soc_component_update_bits_legacy(component, reg,
156 mask, val, &change);
157
158 if (ret < 0)
159 return ret;
160 return change;
161}
162EXPORT_SYMBOL_GPL(snd_soc_component_update_bits_async);
163
164
165
166
167
168
169
170
171void snd_soc_component_async_complete(struct snd_soc_component *component)
172{
173 if (component->regmap)
174 regmap_async_complete(component->regmap);
175}
176EXPORT_SYMBOL_GPL(snd_soc_component_async_complete);
177
178
179
180
181
182
183
184
185
186
187
188
189
190int snd_soc_component_test_bits(struct snd_soc_component *component,
191 unsigned int reg, unsigned int mask, unsigned int value)
192{
193 unsigned int old, new;
194 int ret;
195
196 ret = snd_soc_component_read(component, reg, &old);
197 if (ret < 0)
198 return ret;
199 new = (old & ~mask) | value;
200 return old != new;
201}
202EXPORT_SYMBOL_GPL(snd_soc_component_test_bits);
203