diff options
Diffstat (limited to 'sound/soc/codecs/wm8728.c')
-rw-r--r-- | sound/soc/codecs/wm8728.c | 311 |
1 files changed, 97 insertions, 214 deletions
diff --git a/sound/soc/codecs/wm8728.c b/sound/soc/codecs/wm8728.c index 34be2d2b69ef..86d4718d3a76 100644 --- a/sound/soc/codecs/wm8728.c +++ b/sound/soc/codecs/wm8728.c | |||
@@ -23,14 +23,11 @@ | |||
23 | #include <sound/pcm.h> | 23 | #include <sound/pcm.h> |
24 | #include <sound/pcm_params.h> | 24 | #include <sound/pcm_params.h> |
25 | #include <sound/soc.h> | 25 | #include <sound/soc.h> |
26 | #include <sound/soc-dapm.h> | ||
27 | #include <sound/initval.h> | 26 | #include <sound/initval.h> |
28 | #include <sound/tlv.h> | 27 | #include <sound/tlv.h> |
29 | 28 | ||
30 | #include "wm8728.h" | 29 | #include "wm8728.h" |
31 | 30 | ||
32 | struct snd_soc_codec_device soc_codec_dev_wm8728; | ||
33 | |||
34 | /* | 31 | /* |
35 | * We can't read the WM8728 register space so we cache them instead. | 32 | * We can't read the WM8728 register space so we cache them instead. |
36 | * Note that the defaults here aren't the physical defaults, we latch | 33 | * Note that the defaults here aren't the physical defaults, we latch |
@@ -44,6 +41,11 @@ static const u16 wm8728_reg_defaults[] = { | |||
44 | 0x100, | 41 | 0x100, |
45 | }; | 42 | }; |
46 | 43 | ||
44 | /* codec private data */ | ||
45 | struct wm8728_priv { | ||
46 | enum snd_soc_control_type control_type; | ||
47 | }; | ||
48 | |||
47 | static const DECLARE_TLV_DB_SCALE(wm8728_tlv, -12750, 50, 1); | 49 | static const DECLARE_TLV_DB_SCALE(wm8728_tlv, -12750, 50, 1); |
48 | 50 | ||
49 | static const struct snd_kcontrol_new wm8728_snd_controls[] = { | 51 | static const struct snd_kcontrol_new wm8728_snd_controls[] = { |
@@ -63,21 +65,11 @@ SND_SOC_DAPM_OUTPUT("VOUTL"), | |||
63 | SND_SOC_DAPM_OUTPUT("VOUTR"), | 65 | SND_SOC_DAPM_OUTPUT("VOUTR"), |
64 | }; | 66 | }; |
65 | 67 | ||
66 | static const struct snd_soc_dapm_route intercon[] = { | 68 | static const struct snd_soc_dapm_route wm8728_intercon[] = { |
67 | {"VOUTL", NULL, "DAC"}, | 69 | {"VOUTL", NULL, "DAC"}, |
68 | {"VOUTR", NULL, "DAC"}, | 70 | {"VOUTR", NULL, "DAC"}, |
69 | }; | 71 | }; |
70 | 72 | ||
71 | static int wm8728_add_widgets(struct snd_soc_codec *codec) | ||
72 | { | ||
73 | snd_soc_dapm_new_controls(codec, wm8728_dapm_widgets, | ||
74 | ARRAY_SIZE(wm8728_dapm_widgets)); | ||
75 | |||
76 | snd_soc_dapm_add_routes(codec, intercon, ARRAY_SIZE(intercon)); | ||
77 | |||
78 | return 0; | ||
79 | } | ||
80 | |||
81 | static int wm8728_mute(struct snd_soc_dai *dai, int mute) | 73 | static int wm8728_mute(struct snd_soc_dai *dai, int mute) |
82 | { | 74 | { |
83 | struct snd_soc_codec *codec = dai->codec; | 75 | struct snd_soc_codec *codec = dai->codec; |
@@ -96,8 +88,7 @@ static int wm8728_hw_params(struct snd_pcm_substream *substream, | |||
96 | struct snd_soc_dai *dai) | 88 | struct snd_soc_dai *dai) |
97 | { | 89 | { |
98 | struct snd_soc_pcm_runtime *rtd = substream->private_data; | 90 | struct snd_soc_pcm_runtime *rtd = substream->private_data; |
99 | struct snd_soc_device *socdev = rtd->socdev; | 91 | struct snd_soc_codec *codec = rtd->codec; |
100 | struct snd_soc_codec *codec = socdev->card->codec; | ||
101 | u16 dac = snd_soc_read(codec, WM8728_DACCTL); | 92 | u16 dac = snd_soc_read(codec, WM8728_DACCTL); |
102 | 93 | ||
103 | dac &= ~0x18; | 94 | dac &= ~0x18; |
@@ -178,7 +169,7 @@ static int wm8728_set_bias_level(struct snd_soc_codec *codec, | |||
178 | case SND_SOC_BIAS_ON: | 169 | case SND_SOC_BIAS_ON: |
179 | case SND_SOC_BIAS_PREPARE: | 170 | case SND_SOC_BIAS_PREPARE: |
180 | case SND_SOC_BIAS_STANDBY: | 171 | case SND_SOC_BIAS_STANDBY: |
181 | if (codec->bias_level == SND_SOC_BIAS_OFF) { | 172 | if (codec->dapm.bias_level == SND_SOC_BIAS_OFF) { |
182 | /* Power everything up... */ | 173 | /* Power everything up... */ |
183 | reg = snd_soc_read(codec, WM8728_DACCTL); | 174 | reg = snd_soc_read(codec, WM8728_DACCTL); |
184 | snd_soc_write(codec, WM8728_DACCTL, reg & ~0x4); | 175 | snd_soc_write(codec, WM8728_DACCTL, reg & ~0x4); |
@@ -195,7 +186,7 @@ static int wm8728_set_bias_level(struct snd_soc_codec *codec, | |||
195 | snd_soc_write(codec, WM8728_DACCTL, reg | 0x4); | 186 | snd_soc_write(codec, WM8728_DACCTL, reg | 0x4); |
196 | break; | 187 | break; |
197 | } | 188 | } |
198 | codec->bias_level = level; | 189 | codec->dapm.bias_level = level; |
199 | return 0; | 190 | return 0; |
200 | } | 191 | } |
201 | 192 | ||
@@ -210,8 +201,8 @@ static struct snd_soc_dai_ops wm8728_dai_ops = { | |||
210 | .set_fmt = wm8728_set_dai_fmt, | 201 | .set_fmt = wm8728_set_dai_fmt, |
211 | }; | 202 | }; |
212 | 203 | ||
213 | struct snd_soc_dai wm8728_dai = { | 204 | static struct snd_soc_dai_driver wm8728_dai = { |
214 | .name = "WM8728", | 205 | .name = "wm8728-hifi", |
215 | .playback = { | 206 | .playback = { |
216 | .stream_name = "Playback", | 207 | .stream_name = "Playback", |
217 | .channels_min = 2, | 208 | .channels_min = 2, |
@@ -221,63 +212,31 @@ struct snd_soc_dai wm8728_dai = { | |||
221 | }, | 212 | }, |
222 | .ops = &wm8728_dai_ops, | 213 | .ops = &wm8728_dai_ops, |
223 | }; | 214 | }; |
224 | EXPORT_SYMBOL_GPL(wm8728_dai); | ||
225 | 215 | ||
226 | static int wm8728_suspend(struct platform_device *pdev, pm_message_t state) | 216 | static int wm8728_suspend(struct snd_soc_codec *codec, pm_message_t state) |
227 | { | 217 | { |
228 | struct snd_soc_device *socdev = platform_get_drvdata(pdev); | ||
229 | struct snd_soc_codec *codec = socdev->card->codec; | ||
230 | |||
231 | wm8728_set_bias_level(codec, SND_SOC_BIAS_OFF); | 218 | wm8728_set_bias_level(codec, SND_SOC_BIAS_OFF); |
232 | 219 | ||
233 | return 0; | 220 | return 0; |
234 | } | 221 | } |
235 | 222 | ||
236 | static int wm8728_resume(struct platform_device *pdev) | 223 | static int wm8728_resume(struct snd_soc_codec *codec) |
237 | { | 224 | { |
238 | struct snd_soc_device *socdev = platform_get_drvdata(pdev); | ||
239 | struct snd_soc_codec *codec = socdev->card->codec; | ||
240 | |||
241 | wm8728_set_bias_level(codec, SND_SOC_BIAS_STANDBY); | 225 | wm8728_set_bias_level(codec, SND_SOC_BIAS_STANDBY); |
242 | 226 | ||
243 | return 0; | 227 | return 0; |
244 | } | 228 | } |
245 | 229 | ||
246 | /* | 230 | static int wm8728_probe(struct snd_soc_codec *codec) |
247 | * initialise the WM8728 driver | ||
248 | * register the mixer and dsp interfaces with the kernel | ||
249 | */ | ||
250 | static int wm8728_init(struct snd_soc_device *socdev, | ||
251 | enum snd_soc_control_type control) | ||
252 | { | 231 | { |
253 | struct snd_soc_codec *codec = socdev->card->codec; | 232 | struct wm8728_priv *wm8728 = snd_soc_codec_get_drvdata(codec); |
254 | int ret = 0; | 233 | int ret; |
255 | |||
256 | codec->name = "WM8728"; | ||
257 | codec->owner = THIS_MODULE; | ||
258 | codec->set_bias_level = wm8728_set_bias_level; | ||
259 | codec->dai = &wm8728_dai; | ||
260 | codec->num_dai = 1; | ||
261 | codec->bias_level = SND_SOC_BIAS_OFF; | ||
262 | codec->reg_cache_size = ARRAY_SIZE(wm8728_reg_defaults); | ||
263 | codec->reg_cache = kmemdup(wm8728_reg_defaults, | ||
264 | sizeof(wm8728_reg_defaults), | ||
265 | GFP_KERNEL); | ||
266 | if (codec->reg_cache == NULL) | ||
267 | return -ENOMEM; | ||
268 | 234 | ||
269 | ret = snd_soc_codec_set_cache_io(codec, 7, 9, control); | 235 | ret = snd_soc_codec_set_cache_io(codec, 7, 9, wm8728->control_type); |
270 | if (ret < 0) { | 236 | if (ret < 0) { |
271 | printk(KERN_ERR "wm8728: failed to configure cache I/O: %d\n", | 237 | printk(KERN_ERR "wm8728: failed to configure cache I/O: %d\n", |
272 | ret); | 238 | ret); |
273 | goto err; | 239 | return ret; |
274 | } | ||
275 | |||
276 | /* register pcms */ | ||
277 | ret = snd_soc_new_pcms(socdev, SNDRV_DEFAULT_IDX1, SNDRV_DEFAULT_STR1); | ||
278 | if (ret < 0) { | ||
279 | printk(KERN_ERR "wm8728: failed to create pcms\n"); | ||
280 | goto err; | ||
281 | } | 240 | } |
282 | 241 | ||
283 | /* power on device */ | 242 | /* power on device */ |
@@ -285,218 +244,142 @@ static int wm8728_init(struct snd_soc_device *socdev, | |||
285 | 244 | ||
286 | snd_soc_add_controls(codec, wm8728_snd_controls, | 245 | snd_soc_add_controls(codec, wm8728_snd_controls, |
287 | ARRAY_SIZE(wm8728_snd_controls)); | 246 | ARRAY_SIZE(wm8728_snd_controls)); |
288 | wm8728_add_widgets(codec); | ||
289 | |||
290 | return ret; | ||
291 | 247 | ||
292 | err: | ||
293 | kfree(codec->reg_cache); | ||
294 | return ret; | 248 | return ret; |
295 | } | 249 | } |
296 | 250 | ||
297 | static struct snd_soc_device *wm8728_socdev; | 251 | static int wm8728_remove(struct snd_soc_codec *codec) |
298 | 252 | { | |
299 | #if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE) | 253 | wm8728_set_bias_level(codec, SND_SOC_BIAS_OFF); |
254 | return 0; | ||
255 | } | ||
300 | 256 | ||
301 | /* | 257 | static struct snd_soc_codec_driver soc_codec_dev_wm8728 = { |
302 | * WM8728 2 wire address is determined by GPIO5 | 258 | .probe = wm8728_probe, |
303 | * state during powerup. | 259 | .remove = wm8728_remove, |
304 | * low = 0x1a | 260 | .suspend = wm8728_suspend, |
305 | * high = 0x1b | 261 | .resume = wm8728_resume, |
306 | */ | 262 | .set_bias_level = wm8728_set_bias_level, |
263 | .reg_cache_size = ARRAY_SIZE(wm8728_reg_defaults), | ||
264 | .reg_word_size = sizeof(u16), | ||
265 | .reg_cache_default = wm8728_reg_defaults, | ||
266 | .dapm_widgets = wm8728_dapm_widgets, | ||
267 | .num_dapm_widgets = ARRAY_SIZE(wm8728_dapm_widgets), | ||
268 | .dapm_routes = wm8728_intercon, | ||
269 | .num_dapm_routes = ARRAY_SIZE(wm8728_intercon), | ||
270 | }; | ||
307 | 271 | ||
308 | static int wm8728_i2c_probe(struct i2c_client *i2c, | 272 | #if defined(CONFIG_SPI_MASTER) |
309 | const struct i2c_device_id *id) | 273 | static int __devinit wm8728_spi_probe(struct spi_device *spi) |
310 | { | 274 | { |
311 | struct snd_soc_device *socdev = wm8728_socdev; | 275 | struct wm8728_priv *wm8728; |
312 | struct snd_soc_codec *codec = socdev->card->codec; | ||
313 | int ret; | 276 | int ret; |
314 | 277 | ||
315 | i2c_set_clientdata(i2c, codec); | 278 | wm8728 = kzalloc(sizeof(struct wm8728_priv), GFP_KERNEL); |
316 | codec->control_data = i2c; | 279 | if (wm8728 == NULL) |
280 | return -ENOMEM; | ||
281 | |||
282 | wm8728->control_type = SND_SOC_SPI; | ||
283 | spi_set_drvdata(spi, wm8728); | ||
317 | 284 | ||
318 | ret = wm8728_init(socdev, SND_SOC_I2C); | 285 | ret = snd_soc_register_codec(&spi->dev, |
286 | &soc_codec_dev_wm8728, &wm8728_dai, 1); | ||
319 | if (ret < 0) | 287 | if (ret < 0) |
320 | pr_err("failed to initialise WM8728\n"); | 288 | kfree(wm8728); |
321 | |||
322 | return ret; | 289 | return ret; |
323 | } | 290 | } |
324 | 291 | ||
325 | static int wm8728_i2c_remove(struct i2c_client *client) | 292 | static int __devexit wm8728_spi_remove(struct spi_device *spi) |
326 | { | 293 | { |
327 | struct snd_soc_codec *codec = i2c_get_clientdata(client); | 294 | snd_soc_unregister_codec(&spi->dev); |
328 | kfree(codec->reg_cache); | 295 | kfree(spi_get_drvdata(spi)); |
329 | return 0; | 296 | return 0; |
330 | } | 297 | } |
331 | 298 | ||
332 | static const struct i2c_device_id wm8728_i2c_id[] = { | 299 | static struct spi_driver wm8728_spi_driver = { |
333 | { "wm8728", 0 }, | ||
334 | { } | ||
335 | }; | ||
336 | MODULE_DEVICE_TABLE(i2c, wm8728_i2c_id); | ||
337 | |||
338 | static struct i2c_driver wm8728_i2c_driver = { | ||
339 | .driver = { | 300 | .driver = { |
340 | .name = "WM8728 I2C Codec", | 301 | .name = "wm8728-codec", |
341 | .owner = THIS_MODULE, | 302 | .owner = THIS_MODULE, |
342 | }, | 303 | }, |
343 | .probe = wm8728_i2c_probe, | 304 | .probe = wm8728_spi_probe, |
344 | .remove = wm8728_i2c_remove, | 305 | .remove = __devexit_p(wm8728_spi_remove), |
345 | .id_table = wm8728_i2c_id, | ||
346 | }; | 306 | }; |
307 | #endif /* CONFIG_SPI_MASTER */ | ||
347 | 308 | ||
348 | static int wm8728_add_i2c_device(struct platform_device *pdev, | 309 | #if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE) |
349 | const struct wm8728_setup_data *setup) | 310 | static __devinit int wm8728_i2c_probe(struct i2c_client *i2c, |
311 | const struct i2c_device_id *id) | ||
350 | { | 312 | { |
351 | struct i2c_board_info info; | 313 | struct wm8728_priv *wm8728; |
352 | struct i2c_adapter *adapter; | ||
353 | struct i2c_client *client; | ||
354 | int ret; | 314 | int ret; |
355 | 315 | ||
356 | ret = i2c_add_driver(&wm8728_i2c_driver); | 316 | wm8728 = kzalloc(sizeof(struct wm8728_priv), GFP_KERNEL); |
357 | if (ret != 0) { | 317 | if (wm8728 == NULL) |
358 | dev_err(&pdev->dev, "can't add i2c driver\n"); | 318 | return -ENOMEM; |
359 | return ret; | ||
360 | } | ||
361 | |||
362 | memset(&info, 0, sizeof(struct i2c_board_info)); | ||
363 | info.addr = setup->i2c_address; | ||
364 | strlcpy(info.type, "wm8728", I2C_NAME_SIZE); | ||
365 | |||
366 | adapter = i2c_get_adapter(setup->i2c_bus); | ||
367 | if (!adapter) { | ||
368 | dev_err(&pdev->dev, "can't get i2c adapter %d\n", | ||
369 | setup->i2c_bus); | ||
370 | goto err_driver; | ||
371 | } | ||
372 | |||
373 | client = i2c_new_device(adapter, &info); | ||
374 | i2c_put_adapter(adapter); | ||
375 | if (!client) { | ||
376 | dev_err(&pdev->dev, "can't add i2c device at 0x%x\n", | ||
377 | (unsigned int)info.addr); | ||
378 | goto err_driver; | ||
379 | } | ||
380 | |||
381 | return 0; | ||
382 | |||
383 | err_driver: | ||
384 | i2c_del_driver(&wm8728_i2c_driver); | ||
385 | return -ENODEV; | ||
386 | } | ||
387 | #endif | ||
388 | |||
389 | #if defined(CONFIG_SPI_MASTER) | ||
390 | static int __devinit wm8728_spi_probe(struct spi_device *spi) | ||
391 | { | ||
392 | struct snd_soc_device *socdev = wm8728_socdev; | ||
393 | struct snd_soc_codec *codec = socdev->card->codec; | ||
394 | int ret; | ||
395 | 319 | ||
396 | codec->control_data = spi; | 320 | i2c_set_clientdata(i2c, wm8728); |
321 | wm8728->control_type = SND_SOC_I2C; | ||
397 | 322 | ||
398 | ret = wm8728_init(socdev, SND_SOC_SPI); | 323 | ret = snd_soc_register_codec(&i2c->dev, |
324 | &soc_codec_dev_wm8728, &wm8728_dai, 1); | ||
399 | if (ret < 0) | 325 | if (ret < 0) |
400 | dev_err(&spi->dev, "failed to initialise WM8728\n"); | 326 | kfree(wm8728); |
401 | |||
402 | return ret; | 327 | return ret; |
403 | } | 328 | } |
404 | 329 | ||
405 | static int __devexit wm8728_spi_remove(struct spi_device *spi) | 330 | static __devexit int wm8728_i2c_remove(struct i2c_client *client) |
406 | { | 331 | { |
332 | snd_soc_unregister_codec(&client->dev); | ||
333 | kfree(i2c_get_clientdata(client)); | ||
407 | return 0; | 334 | return 0; |
408 | } | 335 | } |
409 | 336 | ||
410 | static struct spi_driver wm8728_spi_driver = { | 337 | static const struct i2c_device_id wm8728_i2c_id[] = { |
338 | { "wm8728", 0 }, | ||
339 | { } | ||
340 | }; | ||
341 | MODULE_DEVICE_TABLE(i2c, wm8728_i2c_id); | ||
342 | |||
343 | static struct i2c_driver wm8728_i2c_driver = { | ||
411 | .driver = { | 344 | .driver = { |
412 | .name = "wm8728", | 345 | .name = "wm8728-codec", |
413 | .bus = &spi_bus_type, | 346 | .owner = THIS_MODULE, |
414 | .owner = THIS_MODULE, | ||
415 | }, | 347 | }, |
416 | .probe = wm8728_spi_probe, | 348 | .probe = wm8728_i2c_probe, |
417 | .remove = __devexit_p(wm8728_spi_remove), | 349 | .remove = __devexit_p(wm8728_i2c_remove), |
350 | .id_table = wm8728_i2c_id, | ||
418 | }; | 351 | }; |
419 | #endif /* CONFIG_SPI_MASTER */ | 352 | #endif |
420 | 353 | ||
421 | static int wm8728_probe(struct platform_device *pdev) | 354 | static int __init wm8728_modinit(void) |
422 | { | 355 | { |
423 | struct snd_soc_device *socdev = platform_get_drvdata(pdev); | ||
424 | struct wm8728_setup_data *setup; | ||
425 | struct snd_soc_codec *codec; | ||
426 | int ret = 0; | 356 | int ret = 0; |
427 | |||
428 | setup = socdev->codec_data; | ||
429 | codec = kzalloc(sizeof(struct snd_soc_codec), GFP_KERNEL); | ||
430 | if (codec == NULL) | ||
431 | return -ENOMEM; | ||
432 | |||
433 | socdev->card->codec = codec; | ||
434 | mutex_init(&codec->mutex); | ||
435 | INIT_LIST_HEAD(&codec->dapm_widgets); | ||
436 | INIT_LIST_HEAD(&codec->dapm_paths); | ||
437 | |||
438 | wm8728_socdev = socdev; | ||
439 | ret = -ENODEV; | ||
440 | |||
441 | #if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE) | 357 | #if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE) |
442 | if (setup->i2c_address) { | 358 | ret = i2c_add_driver(&wm8728_i2c_driver); |
443 | ret = wm8728_add_i2c_device(pdev, setup); | 359 | if (ret != 0) { |
360 | printk(KERN_ERR "Failed to register wm8728 I2C driver: %d\n", | ||
361 | ret); | ||
444 | } | 362 | } |
445 | #endif | 363 | #endif |
446 | #if defined(CONFIG_SPI_MASTER) | 364 | #if defined(CONFIG_SPI_MASTER) |
447 | if (setup->spi) { | 365 | ret = spi_register_driver(&wm8728_spi_driver); |
448 | ret = spi_register_driver(&wm8728_spi_driver); | 366 | if (ret != 0) { |
449 | if (ret != 0) | 367 | printk(KERN_ERR "Failed to register wm8728 SPI driver: %d\n", |
450 | printk(KERN_ERR "can't add spi driver"); | 368 | ret); |
451 | } | 369 | } |
452 | #endif | 370 | #endif |
453 | |||
454 | if (ret != 0) | ||
455 | kfree(codec); | ||
456 | |||
457 | return ret; | 371 | return ret; |
458 | } | 372 | } |
373 | module_init(wm8728_modinit); | ||
459 | 374 | ||
460 | /* power down chip */ | 375 | static void __exit wm8728_exit(void) |
461 | static int wm8728_remove(struct platform_device *pdev) | ||
462 | { | 376 | { |
463 | struct snd_soc_device *socdev = platform_get_drvdata(pdev); | ||
464 | struct snd_soc_codec *codec = socdev->card->codec; | ||
465 | |||
466 | if (codec->control_data) | ||
467 | wm8728_set_bias_level(codec, SND_SOC_BIAS_OFF); | ||
468 | |||
469 | snd_soc_free_pcms(socdev); | ||
470 | snd_soc_dapm_free(socdev); | ||
471 | #if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE) | 377 | #if defined(CONFIG_I2C) || defined(CONFIG_I2C_MODULE) |
472 | i2c_unregister_device(codec->control_data); | ||
473 | i2c_del_driver(&wm8728_i2c_driver); | 378 | i2c_del_driver(&wm8728_i2c_driver); |
474 | #endif | 379 | #endif |
475 | #if defined(CONFIG_SPI_MASTER) | 380 | #if defined(CONFIG_SPI_MASTER) |
476 | spi_unregister_driver(&wm8728_spi_driver); | 381 | spi_unregister_driver(&wm8728_spi_driver); |
477 | #endif | 382 | #endif |
478 | kfree(codec); | ||
479 | |||
480 | return 0; | ||
481 | } | ||
482 | |||
483 | struct snd_soc_codec_device soc_codec_dev_wm8728 = { | ||
484 | .probe = wm8728_probe, | ||
485 | .remove = wm8728_remove, | ||
486 | .suspend = wm8728_suspend, | ||
487 | .resume = wm8728_resume, | ||
488 | }; | ||
489 | EXPORT_SYMBOL_GPL(soc_codec_dev_wm8728); | ||
490 | |||
491 | static int __init wm8728_modinit(void) | ||
492 | { | ||
493 | return snd_soc_register_dai(&wm8728_dai); | ||
494 | } | ||
495 | module_init(wm8728_modinit); | ||
496 | |||
497 | static void __exit wm8728_exit(void) | ||
498 | { | ||
499 | snd_soc_unregister_dai(&wm8728_dai); | ||
500 | } | 383 | } |
501 | module_exit(wm8728_exit); | 384 | module_exit(wm8728_exit); |
502 | 385 | ||