neo1973_wm8753.c 10.7 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23
/*
 * neo1973_wm8753.c  --  SoC audio for Neo1973
 *
 * Copyright 2007 Wolfson Microelectronics PLC.
 * Author: Graeme Gregory
 *         graeme.gregory@wolfsonmicro.com or linux@wolfsonmicro.com
 *
 *  This program is free software; you can redistribute  it and/or modify it
 *  under  the terms of  the GNU General  Public License as published by the
 *  Free Software Foundation;  either version 2 of the  License, or (at your
 *  option) any later version.
 *
 */

#include <linux/module.h>
#include <linux/moduleparam.h>
#include <linux/timer.h>
#include <linux/interrupt.h>
#include <linux/platform_device.h>
#include <sound/core.h>
#include <sound/pcm.h>
#include <sound/soc.h>

24
#include <asm/mach-types.h>
25 26 27
#include <mach/regs-clock.h>
#include <mach/regs-gpio.h>
#include <mach/hardware.h>
28
#include <linux/io.h>
29
#include <mach/spi-gpio.h>
30

B
Ben Dooks 已提交
31
#include <plat/regs-iis.h>
32

33
#include "../codecs/wm8753.h"
J
Jassi Brar 已提交
34
#include "dma.h"
35 36
#include "s3c24xx-i2s.h"

37
static struct snd_soc_card neo1973;
38 39 40 41 42

static int neo1973_hifi_hw_params(struct snd_pcm_substream *substream,
	struct snd_pcm_hw_params *params)
{
	struct snd_soc_pcm_runtime *rtd = substream->private_data;
43 44
	struct snd_soc_dai *codec_dai = rtd->codec_dai;
	struct snd_soc_dai *cpu_dai = rtd->cpu_dai;
45 46 47 48
	unsigned int pll_out = 0, bclk = 0;
	int ret = 0;
	unsigned long iis_clkrate;

49
	pr_debug("Entered %s\n", __func__);
50

51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84
	iis_clkrate = s3c24xx_i2s_get_clockrate();

	switch (params_rate(params)) {
	case 8000:
	case 16000:
		pll_out = 12288000;
		break;
	case 48000:
		bclk = WM8753_BCLK_DIV_4;
		pll_out = 12288000;
		break;
	case 96000:
		bclk = WM8753_BCLK_DIV_2;
		pll_out = 12288000;
		break;
	case 11025:
		bclk = WM8753_BCLK_DIV_16;
		pll_out = 11289600;
		break;
	case 22050:
		bclk = WM8753_BCLK_DIV_8;
		pll_out = 11289600;
		break;
	case 44100:
		bclk = WM8753_BCLK_DIV_4;
		pll_out = 11289600;
		break;
	case 88200:
		bclk = WM8753_BCLK_DIV_2;
		pll_out = 11289600;
		break;
	}

	/* set codec DAI configuration */
85
	ret = snd_soc_dai_set_fmt(codec_dai,
86 87 88 89 90 91
		SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_NB_NF |
		SND_SOC_DAIFMT_CBM_CFM);
	if (ret < 0)
		return ret;

	/* set cpu DAI configuration */
92
	ret = snd_soc_dai_set_fmt(cpu_dai,
93 94 95 96 97 98
		SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_NB_NF |
		SND_SOC_DAIFMT_CBM_CFM);
	if (ret < 0)
		return ret;

	/* set the codec system clock for DAC and ADC */
99
	ret = snd_soc_dai_set_sysclk(codec_dai, WM8753_MCLK, pll_out,
100 101 102 103 104
		SND_SOC_CLOCK_IN);
	if (ret < 0)
		return ret;

	/* set MCLK division for sample rate */
105
	ret = snd_soc_dai_set_clkdiv(cpu_dai, S3C24XX_DIV_MCLK,
106
		S3C2410_IISMOD_32FS);
107 108 109 110
	if (ret < 0)
		return ret;

	/* set codec BCLK division for sample rate */
111
	ret = snd_soc_dai_set_clkdiv(codec_dai, WM8753_BCLKDIV, bclk);
112 113 114 115
	if (ret < 0)
		return ret;

	/* set prescaler division for sample rate */
116
	ret = snd_soc_dai_set_clkdiv(cpu_dai, S3C24XX_DIV_PRESCALER,
117
		S3C24XX_PRESCALE(4, 4));
118 119 120 121
	if (ret < 0)
		return ret;

	/* codec PLL input is PCLK/4 */
122
	ret = snd_soc_dai_set_pll(codec_dai, WM8753_PLL1, 0,
123 124 125 126 127 128 129 130 131 132
		iis_clkrate / 4, pll_out);
	if (ret < 0)
		return ret;

	return 0;
}

static int neo1973_hifi_hw_free(struct snd_pcm_substream *substream)
{
	struct snd_soc_pcm_runtime *rtd = substream->private_data;
133
	struct snd_soc_dai *codec_dai = rtd->codec_dai;
134

135
	pr_debug("Entered %s\n", __func__);
136

137
	/* disable the PLL */
138
	return snd_soc_dai_set_pll(codec_dai, WM8753_PLL1, 0, 0, 0);
139 140 141 142 143 144 145 146 147 148 149 150 151 152
}

/*
 * Neo1973 WM8753 HiFi DAI opserations.
 */
static struct snd_soc_ops neo1973_hifi_ops = {
	.hw_params = neo1973_hifi_hw_params,
	.hw_free = neo1973_hifi_hw_free,
};

static int neo1973_voice_hw_params(struct snd_pcm_substream *substream,
	struct snd_pcm_hw_params *params)
{
	struct snd_soc_pcm_runtime *rtd = substream->private_data;
153
	struct snd_soc_dai *codec_dai = rtd->codec_dai;
154 155 156 157
	unsigned int pcmdiv = 0;
	int ret = 0;
	unsigned long iis_clkrate;

158
	pr_debug("Entered %s\n", __func__);
159

160 161 162 163 164 165 166 167 168 169 170
	iis_clkrate = s3c24xx_i2s_get_clockrate();

	if (params_rate(params) != 8000)
		return -EINVAL;
	if (params_channels(params) != 1)
		return -EINVAL;

	pcmdiv = WM8753_PCM_DIV_6; /* 2.048 MHz */

	/* todo: gg check mode (DSP_B) against CSR datasheet */
	/* set codec DAI configuration */
171
	ret = snd_soc_dai_set_fmt(codec_dai, SND_SOC_DAIFMT_DSP_B |
172 173 174 175 176
		SND_SOC_DAIFMT_NB_NF | SND_SOC_DAIFMT_CBS_CFS);
	if (ret < 0)
		return ret;

	/* set the codec system clock for DAC and ADC */
177
	ret = snd_soc_dai_set_sysclk(codec_dai, WM8753_PCMCLK, 12288000,
178 179 180 181 182
		SND_SOC_CLOCK_IN);
	if (ret < 0)
		return ret;

	/* set codec PCM division for sample rate */
183
	ret = snd_soc_dai_set_clkdiv(codec_dai, WM8753_PCMDIV, pcmdiv);
184 185 186
	if (ret < 0)
		return ret;

A
Andrea Gelmini 已提交
187
	/* configure and enable PLL for 12.288MHz output */
188
	ret = snd_soc_dai_set_pll(codec_dai, WM8753_PLL2, 0,
189 190 191 192 193 194 195 196 197 198
		iis_clkrate / 4, 12288000);
	if (ret < 0)
		return ret;

	return 0;
}

static int neo1973_voice_hw_free(struct snd_pcm_substream *substream)
{
	struct snd_soc_pcm_runtime *rtd = substream->private_data;
199
	struct snd_soc_dai *codec_dai = rtd->codec_dai;
200

201
	pr_debug("Entered %s\n", __func__);
202

203
	/* disable the PLL */
204
	return snd_soc_dai_set_pll(codec_dai, WM8753_PLL2, 0, 0, 0);
205 206 207 208 209 210 211 212 213 214 215 216 217 218 219
}

static struct snd_soc_ops neo1973_voice_ops = {
	.hw_params = neo1973_voice_hw_params,
	.hw_free = neo1973_voice_hw_free,
};

static const struct snd_soc_dapm_widget wm8753_dapm_widgets[] = {
	SND_SOC_DAPM_LINE("GSM Line Out", NULL),
	SND_SOC_DAPM_LINE("GSM Line In", NULL),
	SND_SOC_DAPM_MIC("Headset Mic", NULL),
	SND_SOC_DAPM_MIC("Call Mic", NULL),
};


220
static const struct snd_soc_dapm_route neo1973_wm8753_routes[] = {
221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239
	/* Connections to the GSM Module */
	{"GSM Line Out", NULL, "MONO1"},
	{"GSM Line Out", NULL, "MONO2"},
	{"RXP", NULL, "GSM Line In"},
	{"RXN", NULL, "GSM Line In"},

	/* Connections to Headset */
	{"MIC1", NULL, "Mic Bias"},
	{"Mic Bias", NULL, "Headset Mic"},

	/* Call Mic */
	{"MIC2", NULL, "Mic Bias"},
	{"MIC2N", NULL, "Mic Bias"},
	{"Mic Bias", NULL, "Call Mic"},

	/* Connect the ALC pins */
	{"ACIN", NULL, "ACOP"},
};

240
static const struct snd_kcontrol_new neo1973_wm8753_controls[] = {
241 242 243 244
	SOC_DAPM_PIN_SWITCH("GSM Line Out"),
	SOC_DAPM_PIN_SWITCH("GSM Line In"),
	SOC_DAPM_PIN_SWITCH("Headset Mic"),
	SOC_DAPM_PIN_SWITCH("Call Mic"),
245 246
};

247

248 249 250 251 252
/*
 * This is an example machine initialisation for a wm8753 connected to a
 * neo1973 II. It is missing logic to detect hp/mic insertions and logic
 * to re-route the audio in such an event.
 */
253
static int neo1973_wm8753_init(struct snd_soc_pcm_runtime *rtd)
254
{
255
	struct snd_soc_codec *codec = rtd->codec;
L
Liam Girdwood 已提交
256
	struct snd_soc_dapm_context *dapm = &codec->dapm;
257
	int err;
258

259
	pr_debug("Entered %s\n", __func__);
260

261
	/* set up NC codec pins */
L
Liam Girdwood 已提交
262 263 264 265 266 267
	snd_soc_dapm_nc_pin(dapm, "LOUT2");
	snd_soc_dapm_nc_pin(dapm, "ROUT2");
	snd_soc_dapm_nc_pin(dapm, "OUT3");
	snd_soc_dapm_nc_pin(dapm, "OUT4");
	snd_soc_dapm_nc_pin(dapm, "LINE1");
	snd_soc_dapm_nc_pin(dapm, "LINE2");
268 269

	/* Add neo1973 specific widgets */
L
Liam Girdwood 已提交
270
	snd_soc_dapm_new_controls(dapm, wm8753_dapm_widgets,
271
			ARRAY_SIZE(wm8753_dapm_widgets));
272

273
	/* set endpoints to default mode */
274 275 276 277
	snd_soc_dapm_disable_pin(dapm, "GSM Line Out");
	snd_soc_dapm_disable_pin(dapm, "GSM Line In");
	snd_soc_dapm_disable_pin(dapm, "Headset Mic");
	snd_soc_dapm_disable_pin(dapm, "Call Mic");
278

279
	/* add neo1973 specific controls */
280 281
	err = snd_soc_add_controls(codec, neo1973_wm8753_controls,
				ARRAY_SIZE(neo1973_wm8753_controls));
282 283
	if (err < 0)
		return err;
284

285
	/* set up neo1973 specific audio routes */
286 287
	err = snd_soc_dapm_add_routes(dapm, neo1973_wm8753_routes,
			ARRAY_SIZE(neo1973_wm8753_routes));
288

L
Liam Girdwood 已提交
289
	snd_soc_dapm_sync(dapm);
290 291 292
	return 0;
}

293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330
static const struct snd_soc_dapm_route neo1973_lm4857_routes[] = {
	{"Amp IN", NULL, "ROUT1"},
	{"Amp IN", NULL, "LOUT1"},

	{"Handset Spk", NULL, "Amp EP"},
	{"Stereo Out", NULL, "Amp LS"},
	{"Headphone", NULL, "Amp HP"},
};

static const struct snd_soc_dapm_widget neo1973_lm4857_dapm_widgets[] = {
	SND_SOC_DAPM_SPK("Handset Spk", NULL),
	SND_SOC_DAPM_SPK("Stereo Out", NULL),
	SND_SOC_DAPM_HP("Headphone", NULL),
};

static int neo1973_lm4857_init(struct snd_soc_dapm_context *dapm)
{
	int ret;

	ret = snd_soc_dapm_new_controls(dapm, neo1973_lm4857_dapm_widgets,
			ARRAY_SIZE(neo1973_lm4857_dapm_widgets));
	if (ret)
		return ret;

	ret = snd_soc_dapm_add_routes(dapm, neo1973_lm4857_routes,
			ARRAY_SIZE(neo1973_lm4857_routes));
	if (ret)
		return ret;

	snd_soc_dapm_ignore_suspend(dapm, "Stereo Out");
	snd_soc_dapm_ignore_suspend(dapm, "Handset Spk");
	snd_soc_dapm_ignore_suspend(dapm, "Headphone");

	snd_soc_dapm_sync(dapm);

	return 0;
}

331 332 333
/*
 * BT Codec DAI
 */
334
static struct snd_soc_dai bt_dai = {
335
	.name = "bluetooth-dai",
336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351
	.playback = {
		.channels_min = 1,
		.channels_max = 1,
		.rates = SNDRV_PCM_RATE_8000,
		.formats = SNDRV_PCM_FMTBIT_S16_LE,},
	.capture = {
		.channels_min = 1,
		.channels_max = 1,
		.rates = SNDRV_PCM_RATE_8000,
		.formats = SNDRV_PCM_FMTBIT_S16_LE,},
};

static struct snd_soc_dai_link neo1973_dai[] = {
{ /* Hifi Playback - for similatious use with voice below */
	.name = "WM8753",
	.stream_name = "WM8753 HiFi",
J
Jassi Brar 已提交
352
	.platform_name = "samsung-audio",
353
	.cpu_dai_name = "s3c24xx-iis",
354
	.codec_dai_name = "wm8753-hifi",
355
	.codec_name = "wm8753-codec.0-001a",
356 357 358 359 360 361
	.init = neo1973_wm8753_init,
	.ops = &neo1973_hifi_ops,
},
{ /* Voice via BT */
	.name = "Bluetooth",
	.stream_name = "Voice",
J
Jassi Brar 已提交
362
	.platform_name = "samsung-audio",
363 364
	.cpu_dai_name = "bluetooth-dai",
	.codec_dai_name = "wm8753-voice",
365
	.codec_name = "wm8753-codec.0-001a",
366 367 368 369
	.ops = &neo1973_voice_ops,
},
};

370 371 372 373 374 375
static struct snd_soc_aux_dev neo1973_aux_devs[] = {
	{
		.name = "lm4857",
		.codec_name = "lm4857.0-007c",
		.init = neo1973_lm4857_init,
	},
376 377
};

378 379 380 381 382
static struct snd_soc_codec_conf neo1973_codec_conf[] = {
	{
		.dev_name = "lm4857.0-007c",
		.name_prefix = "Amp",
	},
383 384
};

385 386 387 388 389 390 391 392
static struct snd_soc_card neo1973 = {
	.name = "neo1973",
	.dai_link = neo1973_dai,
	.num_links = ARRAY_SIZE(neo1973_dai),
	.aux_dev = neo1973_aux_devs,
	.num_aux_devs = ARRAY_SIZE(neo1973_aux_devs),
	.codec_conf = neo1973_codec_conf,
	.num_configs = ARRAY_SIZE(neo1973_codec_conf),
393 394 395 396 397 398 399 400
};

static struct platform_device *neo1973_snd_device;

static int __init neo1973_init(void)
{
	int ret;

401
	pr_debug("Entered %s\n", __func__);
402

403 404 405 406 407 408
	if (!machine_is_neo1973_gta01()) {
		printk(KERN_INFO
			"Only GTA01 hardware supported by ASoC driver\n");
		return -ENODEV;
	}

409 410 411 412
	neo1973_snd_device = platform_device_alloc("soc-audio", -1);
	if (!neo1973_snd_device)
		return -ENOMEM;

413
	platform_set_drvdata(neo1973_snd_device, &neo1973);
414 415
	ret = platform_device_add(neo1973_snd_device);

416
	if (ret) {
417
		platform_device_put(neo1973_snd_device);
418 419
		return ret;
	}
420

421
	if (ret != 0)
422
		platform_device_unregister(neo1973_snd_device);
423 424 425 426 427 428

	return ret;
}

static void __exit neo1973_exit(void)
{
429
	pr_debug("Entered %s\n", __func__);
430

431 432 433 434 435 436 437
	platform_device_unregister(neo1973_snd_device);
}

module_init(neo1973_init);
module_exit(neo1973_exit);

/* Module information */
438
MODULE_AUTHOR("Graeme Gregory, graeme@openmoko.org, www.openmoko.org");
439 440
MODULE_DESCRIPTION("ALSA SoC WM8753 Neo1973");
MODULE_LICENSE("GPL");