neo1973_wm8753.c 13.4 KB
Newer Older
1
/*
2
 * neo1973_wm8753.c  --  SoC audio for Openmoko Neo1973 and Freerunner devices
3
 *
4 5
 * Copyright 2007 Openmoko Inc
 * Author: Graeme Gregory <graeme@openmoko.org>
6 7 8
 * Copyright 2007 Wolfson Microelectronics PLC.
 * Author: Graeme Gregory
 *         graeme.gregory@wolfsonmicro.com or linux@wolfsonmicro.com
9
 * Copyright 2009 Wolfson Microelectronics
10 11 12 13 14 15 16 17 18
 *
 *  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/platform_device.h>
19 20
#include <linux/gpio.h>

21 22
#include <sound/soc.h>

23
#include <asm/mach-types.h>
B
Ben Dooks 已提交
24
#include <plat/regs-iis.h>
25
#include <mach/gta02.h>
26

27 28 29 30 31 32 33
#include "../codecs/wm8753.h"
#include "s3c24xx-i2s.h"

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;
34 35
	struct snd_soc_dai *codec_dai = rtd->codec_dai;
	struct snd_soc_dai *cpu_dai = rtd->cpu_dai;
36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73
	unsigned int pll_out = 0, bclk = 0;
	int ret = 0;
	unsigned long iis_clkrate;

	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 */
74
	ret = snd_soc_dai_set_fmt(codec_dai,
75 76 77 78 79 80
		SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_NB_NF |
		SND_SOC_DAIFMT_CBM_CFM);
	if (ret < 0)
		return ret;

	/* set cpu DAI configuration */
81
	ret = snd_soc_dai_set_fmt(cpu_dai,
82 83 84 85 86 87
		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 */
88
	ret = snd_soc_dai_set_sysclk(codec_dai, WM8753_MCLK, pll_out,
89 90 91 92 93
		SND_SOC_CLOCK_IN);
	if (ret < 0)
		return ret;

	/* set MCLK division for sample rate */
94
	ret = snd_soc_dai_set_clkdiv(cpu_dai, S3C24XX_DIV_MCLK,
95
		S3C2410_IISMOD_32FS);
96 97 98 99
	if (ret < 0)
		return ret;

	/* set codec BCLK division for sample rate */
100
	ret = snd_soc_dai_set_clkdiv(codec_dai, WM8753_BCLKDIV, bclk);
101 102 103 104
	if (ret < 0)
		return ret;

	/* set prescaler division for sample rate */
105
	ret = snd_soc_dai_set_clkdiv(cpu_dai, S3C24XX_DIV_PRESCALER,
106
		S3C24XX_PRESCALE(4, 4));
107 108 109 110
	if (ret < 0)
		return ret;

	/* codec PLL input is PCLK/4 */
111
	ret = snd_soc_dai_set_pll(codec_dai, WM8753_PLL1, 0,
112 113 114 115 116 117 118 119 120 121
		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;
122
	struct snd_soc_dai *codec_dai = rtd->codec_dai;
123 124

	/* disable the PLL */
125
	return snd_soc_dai_set_pll(codec_dai, WM8753_PLL1, 0, 0, 0);
126 127 128 129 130 131 132 133 134 135 136 137 138 139
}

/*
 * 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;
140
	struct snd_soc_dai *codec_dai = rtd->codec_dai;
141 142 143 144 145 146 147 148 149 150 151 152 153 154 155
	unsigned int pcmdiv = 0;
	int ret = 0;
	unsigned long iis_clkrate;

	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 */
156
	ret = snd_soc_dai_set_fmt(codec_dai, SND_SOC_DAIFMT_DSP_B |
157 158 159 160 161
		SND_SOC_DAIFMT_NB_NF | SND_SOC_DAIFMT_CBS_CFS);
	if (ret < 0)
		return ret;

	/* set the codec system clock for DAC and ADC */
162
	ret = snd_soc_dai_set_sysclk(codec_dai, WM8753_PCMCLK, 12288000,
163 164 165 166 167
		SND_SOC_CLOCK_IN);
	if (ret < 0)
		return ret;

	/* set codec PCM division for sample rate */
168
	ret = snd_soc_dai_set_clkdiv(codec_dai, WM8753_PCMDIV, pcmdiv);
169 170 171
	if (ret < 0)
		return ret;

A
Andrea Gelmini 已提交
172
	/* configure and enable PLL for 12.288MHz output */
173
	ret = snd_soc_dai_set_pll(codec_dai, WM8753_PLL2, 0,
174 175 176 177 178 179 180 181 182 183
		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;
184
	struct snd_soc_dai *codec_dai = rtd->codec_dai;
185 186

	/* disable the PLL */
187
	return snd_soc_dai_set_pll(codec_dai, WM8753_PLL2, 0, 0, 0);
188 189 190 191 192 193 194
}

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

195 196 197
/* Shared routes and controls */

static const struct snd_soc_dapm_widget neo1973_wm8753_dapm_widgets[] = {
198 199 200
	SND_SOC_DAPM_LINE("GSM Line Out", NULL),
	SND_SOC_DAPM_LINE("GSM Line In", NULL),
	SND_SOC_DAPM_MIC("Headset Mic", NULL),
201
	SND_SOC_DAPM_MIC("Handset Mic", NULL),
202 203
};

204
static const struct snd_soc_dapm_route neo1973_wm8753_routes[] = {
205 206 207 208 209 210 211 212 213 214 215 216 217
	/* 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"},
218
	{"Mic Bias", NULL, "Handset Mic"},
219 220 221 222 223

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

224
static const struct snd_kcontrol_new neo1973_wm8753_controls[] = {
225 226 227
	SOC_DAPM_PIN_SWITCH("GSM Line Out"),
	SOC_DAPM_PIN_SWITCH("GSM Line In"),
	SOC_DAPM_PIN_SWITCH("Headset Mic"),
228
	SOC_DAPM_PIN_SWITCH("Handset Mic"),
229 230
};

231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316
/* GTA02 specific routes and controlls */

#ifdef CONFIG_MACH_NEO1973_GTA02

static int gta02_speaker_enabled;

static int lm4853_set_spk(struct snd_kcontrol *kcontrol,
	struct snd_ctl_elem_value *ucontrol)
{
	gta02_speaker_enabled = ucontrol->value.integer.value[0];

	gpio_set_value(GTA02_GPIO_HP_IN, !gta02_speaker_enabled);

	return 0;
}

static int lm4853_get_spk(struct snd_kcontrol *kcontrol,
	struct snd_ctl_elem_value *ucontrol)
{
	ucontrol->value.integer.value[0] = gta02_speaker_enabled;
	return 0;
}

static int lm4853_event(struct snd_soc_dapm_widget *w,
			struct snd_kcontrol *k, int event)
{
	gpio_set_value(GTA02_GPIO_AMP_SHUT, SND_SOC_DAPM_EVENT_OFF(event));

	return 0;
}

static const struct snd_soc_dapm_route neo1973_gta02_routes[] = {
	/* Connections to the amp */
	{"Stereo Out", NULL, "LOUT1"},
	{"Stereo Out", NULL, "ROUT1"},

	/* Call Speaker */
	{"Handset Spk", NULL, "LOUT2"},
	{"Handset Spk", NULL, "ROUT2"},
};

static const struct snd_kcontrol_new neo1973_gta02_wm8753_controls[] = {
	SOC_DAPM_PIN_SWITCH("Handset Spk"),
	SOC_DAPM_PIN_SWITCH("Stereo Out"),

	SOC_SINGLE_BOOL_EXT("Amp Spk Switch", 0,
		lm4853_get_spk,
		lm4853_set_spk),
};

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

static int neo1973_gta02_wm8753_init(struct snd_soc_codec *codec)
{
	struct snd_soc_dapm_context *dapm = &codec->dapm;
	int ret;

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

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

	ret = snd_soc_add_controls(codec, neo1973_gta02_wm8753_controls,
			ARRAY_SIZE(neo1973_gta02_wm8753_controls));
	if (ret)
		return ret;

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

	return 0;
}

#else
static int neo1973_gta02_wm8753_init(struct snd_soc_code *codec) { return 0; }
#endif
317

318
static int neo1973_wm8753_init(struct snd_soc_pcm_runtime *rtd)
319
{
320
	struct snd_soc_codec *codec = rtd->codec;
L
Liam Girdwood 已提交
321
	struct snd_soc_dapm_context *dapm = &codec->dapm;
322
	int ret;
323

324
	/* set up NC codec pins */
325 326 327 328
	if (machine_is_neo1973_gta01()) {
		snd_soc_dapm_nc_pin(dapm, "LOUT2");
		snd_soc_dapm_nc_pin(dapm, "ROUT2");
	}
L
Liam Girdwood 已提交
329 330 331 332
	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");
333 334

	/* Add neo1973 specific widgets */
335 336 337 338
	ret = snd_soc_dapm_new_controls(dapm, neo1973_wm8753_dapm_widgets,
			ARRAY_SIZE(neo1973_wm8753_dapm_widgets));
	if (ret)
		return ret;
339

340
	/* add neo1973 specific controls */
341 342 343 344
	ret = snd_soc_add_controls(codec, neo1973_wm8753_controls,
			ARRAY_SIZE(neo1973_wm8753_controls));
	if (ret)
		return ret;
345

346
	/* set up neo1973 specific audio routes */
347
	ret = snd_soc_dapm_add_routes(dapm, neo1973_wm8753_routes,
348
			ARRAY_SIZE(neo1973_wm8753_routes));
349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368
	if (ret)
		return ret;

	/* set endpoints to default off mode */
	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, "Handset Mic");

	/* allow audio paths from the GSM modem to run during suspend */
	snd_soc_dapm_ignore_suspend(dapm, "GSM Line Out");
	snd_soc_dapm_ignore_suspend(dapm, "GSM Line In");
	snd_soc_dapm_ignore_suspend(dapm, "Headset Mic");
	snd_soc_dapm_ignore_suspend(dapm, "Handset Mic");

	if (machine_is_neo1973_gta02()) {
		ret = neo1973_gta02_wm8753_init(codec);
		if (ret)
			return ret;
	}
369

L
Liam Girdwood 已提交
370
	snd_soc_dapm_sync(dapm);
371

372 373 374
	return 0;
}

375 376 377 378
/* GTA01 specific controlls */

#ifdef CONFIG_MACH_NEO1973_GTA01

379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416
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;
}

417 418 419 420
#else
static int neo1973_lm4857_init(struct snd_soc_dapm_context *dapm) { return 0; };
#endif

421 422 423 424
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 已提交
425
	.platform_name = "samsung-audio",
426
	.cpu_dai_name = "s3c24xx-iis",
427
	.codec_dai_name = "wm8753-hifi",
428
	.codec_name = "wm8753-codec.0-001a",
429 430 431 432 433 434
	.init = neo1973_wm8753_init,
	.ops = &neo1973_hifi_ops,
},
{ /* Voice via BT */
	.name = "Bluetooth",
	.stream_name = "Voice",
J
Jassi Brar 已提交
435
	.platform_name = "samsung-audio",
436
	.cpu_dai_name = "dfbmcs320-pcm",
437
	.codec_dai_name = "wm8753-voice",
438
	.codec_name = "wm8753-codec.0-001a",
439 440 441 442
	.ops = &neo1973_voice_ops,
},
};

443
static struct snd_soc_aux_dev neo1973_aux_devs[] = {
444 445 446 447
	{
		.name = "dfbmcs320",
		.codec_name = "dfbmcs320.0",
	},
448 449 450 451 452
	{
		.name = "lm4857",
		.codec_name = "lm4857.0-007c",
		.init = neo1973_lm4857_init,
	},
453 454
};

455 456 457 458 459
static struct snd_soc_codec_conf neo1973_codec_conf[] = {
	{
		.dev_name = "lm4857.0-007c",
		.name_prefix = "Amp",
	},
460 461
};

462 463 464 465 466 467 468 469 470
#ifdef CONFIG_MACH_NEO1973_GTA02
static const struct gpio neo1973_gta02_gpios[] = {
	{ GTA02_GPIO_HP_IN, GPIOF_OUT_INIT_HIGH, "GTA02_HP_IN" },
	{ GTA02_GPIO_AMP_SHUT, GPIOF_OUT_INIT_HIGH, "GTA02_AMP_SHUT" },
};
#else
static const struct gpio neo1973_gta02_gpios[] = {};
#endif

471 472 473 474 475 476 477 478
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),
479 480 481 482 483 484 485 486
};

static struct platform_device *neo1973_snd_device;

static int __init neo1973_init(void)
{
	int ret;

487
	if (!machine_is_neo1973_gta01() && !machine_is_neo1973_gta02())
488
		return -ENODEV;
489 490 491

	if (machine_is_neo1973_gta02()) {
		neo1973.name = "neo1973gta02";
492
		neo1973.num_aux_devs = 1;
493 494 495 496 497

		ret = gpio_request_array(neo1973_gta02_gpios,
				ARRAY_SIZE(neo1973_gta02_gpios));
		if (ret)
			return ret;
498 499
	}

500
	neo1973_snd_device = platform_device_alloc("soc-audio", -1);
501 502 503 504 505
	if (!neo1973_snd_device) {
		ret = -ENOMEM;
		goto err_gpio_free;
	}

506
	platform_set_drvdata(neo1973_snd_device, &neo1973);
507 508
	ret = platform_device_add(neo1973_snd_device);

509
	if (ret)
510
		goto err_put_device;
511

512
	return 0;
513

514 515 516 517 518 519 520
err_put_device:
	platform_device_put(neo1973_snd_device);
err_gpio_free:
	if (machine_is_neo1973_gta02()) {
		gpio_free_array(neo1973_gta02_gpios,
				ARRAY_SIZE(neo1973_gta02_gpios));
	}
521 522
	return ret;
}
523
module_init(neo1973_init);
524 525 526 527 528

static void __exit neo1973_exit(void)
{
	platform_device_unregister(neo1973_snd_device);

529 530 531 532 533
	if (machine_is_neo1973_gta02()) {
		gpio_free_array(neo1973_gta02_gpios,
				ARRAY_SIZE(neo1973_gta02_gpios));
	}
}
534 535 536
module_exit(neo1973_exit);

/* Module information */
537
MODULE_AUTHOR("Graeme Gregory, graeme@openmoko.org, www.openmoko.org");
538
MODULE_DESCRIPTION("ALSA SoC WM8753 Neo1973 and Frerunner");
539
MODULE_LICENSE("GPL");