sst-mfld-platform-pcm.c 20.8 KB
Newer Older
1
/*
2
 *  sst_mfld_platform.c - Intel MID Platform driver
3
 *
V
Vinod Koul 已提交
4
 *  Copyright (C) 2010-2014 Intel Corp
5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23
 *  Author: Vinod Koul <vinod.koul@intel.com>
 *  Author: Harsha Priya <priya.harsha@intel.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; version 2 of the License.
 *
 *  This program is distributed in the hope that it will be useful, but
 *  WITHOUT ANY WARRANTY; without even the implied warranty of
 *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 *  General Public License for more details.
 *
 * ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
 */
#define pr_fmt(fmt) KBUILD_MODNAME ": " fmt

#include <linux/slab.h>
#include <linux/io.h>
24
#include <linux/module.h>
25 26 27 28
#include <sound/core.h>
#include <sound/pcm.h>
#include <sound/pcm_params.h>
#include <sound/soc.h>
29
#include <sound/compress_driver.h>
V
Vinod Koul 已提交
30
#include <asm/platform_sst_audio.h>
31
#include "sst-mfld-platform.h"
V
Vinod Koul 已提交
32
#include "sst-atom-controls.h"
33

34
struct sst_device *sst;
35 36 37 38
static DEFINE_MUTEX(sst_lock);

int sst_register_dsp(struct sst_device *dev)
{
39 40
	if (WARN_ON(!dev))
		return -EINVAL;
41 42 43 44
	if (!try_module_get(dev->dev->driver->owner))
		return -ENODEV;
	mutex_lock(&sst_lock);
	if (sst) {
45
		dev_err(dev->dev, "we already have a device %s\n", sst->name);
46 47 48 49
		module_put(dev->dev->driver->owner);
		mutex_unlock(&sst_lock);
		return -EEXIST;
	}
50
	dev_dbg(dev->dev, "registering device %s\n", dev->name);
51 52 53 54 55 56 57 58
	sst = dev;
	mutex_unlock(&sst_lock);
	return 0;
}
EXPORT_SYMBOL_GPL(sst_register_dsp);

int sst_unregister_dsp(struct sst_device *dev)
{
59 60
	if (WARN_ON(!dev))
		return -EINVAL;
61 62 63 64 65 66 67 68 69 70 71
	if (dev != sst)
		return -EINVAL;

	mutex_lock(&sst_lock);

	if (!sst) {
		mutex_unlock(&sst_lock);
		return -EIO;
	}

	module_put(sst->dev->driver->owner);
72
	dev_dbg(dev->dev, "unreg %s\n", sst->name);
73 74 75 76 77 78
	sst = NULL;
	mutex_unlock(&sst_lock);
	return 0;
}
EXPORT_SYMBOL_GPL(sst_unregister_dsp);

79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95
static struct snd_pcm_hardware sst_platform_pcm_hw = {
	.info =	(SNDRV_PCM_INFO_INTERLEAVED |
			SNDRV_PCM_INFO_DOUBLE |
			SNDRV_PCM_INFO_PAUSE |
			SNDRV_PCM_INFO_RESUME |
			SNDRV_PCM_INFO_MMAP|
			SNDRV_PCM_INFO_MMAP_VALID |
			SNDRV_PCM_INFO_BLOCK_TRANSFER |
			SNDRV_PCM_INFO_SYNC_START),
	.buffer_bytes_max = SST_MAX_BUFFER,
	.period_bytes_min = SST_MIN_PERIOD_BYTES,
	.period_bytes_max = SST_MAX_PERIOD_BYTES,
	.periods_min = SST_MIN_PERIODS,
	.periods_max = SST_MAX_PERIODS,
	.fifo_size = SST_FIFO_SIZE,
};

V
Vinod Koul 已提交
96 97 98 99 100 101 102
static struct sst_dev_stream_map dpcm_strm_map[] = {
	{0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF}, /* Reserved, not in use */
	{MERR_DPCM_AUDIO, 0, SNDRV_PCM_STREAM_PLAYBACK, PIPE_MEDIA1_IN, SST_TASK_ID_MEDIA, 0},
	{MERR_DPCM_COMPR, 0, SNDRV_PCM_STREAM_PLAYBACK, PIPE_MEDIA0_IN, SST_TASK_ID_MEDIA, 0},
	{MERR_DPCM_AUDIO, 0, SNDRV_PCM_STREAM_CAPTURE, PIPE_PCM1_OUT, SST_TASK_ID_MEDIA, 0},
};

103
static int sst_media_digital_mute(struct snd_soc_dai *dai, int mute, int stream)
104
{
105 106 107

	return sst_send_pipe_gains(dai, stream, mute);
}
108

109
/* helper functions */
110
void sst_set_stream_status(struct sst_runtime_stream *stream,
111 112
					int state)
{
113 114
	unsigned long flags;
	spin_lock_irqsave(&stream->status_lock, flags);
115
	stream->stream_status = state;
116
	spin_unlock_irqrestore(&stream->status_lock, flags);
117 118 119 120 121
}

static inline int sst_get_stream_status(struct sst_runtime_stream *stream)
{
	int state;
122
	unsigned long flags;
123

124
	spin_lock_irqsave(&stream->status_lock, flags);
125
	state = stream->stream_status;
126
	spin_unlock_irqrestore(&stream->status_lock, flags);
127 128 129
	return state;
}

130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148
static void sst_fill_alloc_params(struct snd_pcm_substream *substream,
				struct snd_sst_alloc_params_ext *alloc_param)
{
	unsigned int channels;
	snd_pcm_uframes_t period_size;
	ssize_t periodbytes;
	ssize_t buffer_bytes = snd_pcm_lib_buffer_bytes(substream);
	u32 buffer_addr = virt_to_phys(substream->dma_buffer.area);

	channels = substream->runtime->channels;
	period_size = substream->runtime->period_size;
	periodbytes = samples_to_bytes(substream->runtime, period_size);
	alloc_param->ring_buf_info[0].addr = buffer_addr;
	alloc_param->ring_buf_info[0].size = buffer_bytes;
	alloc_param->sg_count = 1;
	alloc_param->reserved = 0;
	alloc_param->frag_size = periodbytes * channels;

}
149
static void sst_fill_pcm_params(struct snd_pcm_substream *substream,
150
				struct snd_sst_stream_params *param)
151
{
152 153 154
	param->uc.pcm_params.num_chan = (u8) substream->runtime->channels;
	param->uc.pcm_params.pcm_wd_sz = substream->runtime->sample_bits;
	param->uc.pcm_params.sfreq = substream->runtime->rate;
155

156 157 158 159 160 161
	/* PCM stream via ALSA interface */
	param->uc.pcm_params.use_offload_path = 0;
	param->uc.pcm_params.reserved2 = 0;
	memset(param->uc.pcm_params.channel_map, 0, sizeof(u8));

}
V
Vinod Koul 已提交
162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179

static int sst_get_stream_mapping(int dev, int sdev, int dir,
	struct sst_dev_stream_map *map, int size)
{
	int i;

	if (map == NULL)
		return -EINVAL;


	/* index 0 is not used in stream map */
	for (i = 1; i < size; i++) {
		if ((map[i].dev_num == dev) && (map[i].direction == dir))
			return i;
	}
	return 0;
}

180
int sst_fill_stream_params(void *substream,
V
Vinod Koul 已提交
181
	const struct sst_data *ctx, struct snd_sst_params *str_params, bool is_compress)
182
{
V
Vinod Koul 已提交
183 184 185
	int map_size;
	int index;
	struct sst_dev_stream_map *map;
186 187 188
	struct snd_pcm_substream *pstream = NULL;
	struct snd_compr_stream *cstream = NULL;

V
Vinod Koul 已提交
189 190 191
	map = ctx->pdata->pdev_strm_map;
	map_size = ctx->pdata->strm_map_size;

192 193 194 195 196 197 198 199
	if (is_compress == true)
		cstream = (struct snd_compr_stream *)substream;
	else
		pstream = (struct snd_pcm_substream *)substream;

	str_params->stream_type = SST_STREAM_TYPE_MUSIC;

	/* For pcm streams */
V
Vinod Koul 已提交
200 201 202 203 204 205 206 207 208 209 210
	if (pstream) {
		index = sst_get_stream_mapping(pstream->pcm->device,
					  pstream->number, pstream->stream,
					  map, map_size);
		if (index <= 0)
			return -EINVAL;

		str_params->stream_id = index;
		str_params->device_type = map[index].device_id;
		str_params->task = map[index].task_id;

211
		str_params->ops = (u8)pstream->stream;
V
Vinod Koul 已提交
212 213 214 215 216 217 218 219 220 221 222
	}

	if (cstream) {
		index = sst_get_stream_mapping(cstream->device->device,
					       0, cstream->direction,
					       map, map_size);
		if (index <= 0)
			return -EINVAL;
		str_params->stream_id = index;
		str_params->device_type = map[index].device_id;
		str_params->task = map[index].task_id;
223

V
Vinod Koul 已提交
224 225
		str_params->ops = (u8)cstream->direction;
	}
226
	return 0;
227 228
}

229
static int sst_platform_alloc_stream(struct snd_pcm_substream *substream,
230
		struct snd_soc_dai *dai)
231 232 233
{
	struct sst_runtime_stream *stream =
			substream->runtime->private_data;
234 235 236 237
	struct snd_sst_stream_params param = {{{0,},},};
	struct snd_sst_params str_params = {0};
	struct snd_sst_alloc_params_ext alloc_params = {0};
	int ret_val = 0;
238
	struct sst_data *ctx = snd_soc_dai_get_drvdata(dai);
239 240

	/* set codec params and inform SST driver the same */
241
	sst_fill_pcm_params(substream, &param);
242
	sst_fill_alloc_params(substream, &alloc_params);
243 244
	substream->runtime->dma_area = substream->dma_buffer.area;
	str_params.sparams = param;
245 246 247 248
	str_params.aparams = alloc_params;
	str_params.codec = SST_CODEC_TYPE_PCM;

	/* fill the device type and stream id to pass to SST driver */
V
Vinod Koul 已提交
249
	ret_val = sst_fill_stream_params(substream, ctx, &str_params, false);
250 251 252
	if (ret_val < 0)
		return ret_val;

253 254
	stream->stream_info.str_id = str_params.stream_id;

255
	ret_val = stream->ops->open(sst->dev, &str_params);
256 257 258 259
	if (ret_val <= 0)
		return ret_val;


260 261 262
	return ret_val;
}

263
static void sst_period_elapsed(void *arg)
264
{
265
	struct snd_pcm_substream *substream = arg;
266
	struct sst_runtime_stream *stream;
267
	int status;
268 269 270 271 272 273

	if (!substream || !substream->runtime)
		return;
	stream = substream->runtime->private_data;
	if (!stream)
		return;
274 275
	status = sst_get_stream_status(stream);
	if (status != SST_PLATFORM_RUNNING)
276 277 278 279 280 281 282 283
		return;
	snd_pcm_period_elapsed(substream);
}

static int sst_platform_init_stream(struct snd_pcm_substream *substream)
{
	struct sst_runtime_stream *stream =
			substream->runtime->private_data;
284
	struct snd_soc_pcm_runtime *rtd = substream->private_data;
285 286
	int ret_val;

287
	dev_dbg(rtd->dev, "setting buffer ptr param\n");
288
	sst_set_stream_status(stream, SST_PLATFORM_INIT);
289
	stream->stream_info.period_elapsed = sst_period_elapsed;
290
	stream->stream_info.arg = substream;
291 292
	stream->stream_info.buffer_ptr = 0;
	stream->stream_info.sfreq = substream->runtime->rate;
293
	ret_val = stream->ops->stream_init(sst->dev, &stream->stream_info);
294
	if (ret_val)
295
		dev_err(rtd->dev, "control_set ret error %d\n", ret_val);
296 297 298 299
	return ret_val;

}

300 301 302 303 304 305 306 307 308 309
static int power_up_sst(struct sst_runtime_stream *stream)
{
	return stream->ops->power(sst->dev, true);
}

static void power_down_sst(struct sst_runtime_stream *stream)
{
	stream->ops->power(sst->dev, false);
}

310 311
static int sst_media_open(struct snd_pcm_substream *substream,
		struct snd_soc_dai *dai)
312
{
313
	int ret_val = 0;
314
	struct snd_pcm_runtime *runtime = substream->runtime;
315
	struct sst_runtime_stream *stream;
316

317 318 319 320
	stream = kzalloc(sizeof(*stream), GFP_KERNEL);
	if (!stream)
		return -ENOMEM;
	spin_lock_init(&stream->status_lock);
321 322 323

	/* get the sst ops */
	mutex_lock(&sst_lock);
324 325
	if (!sst ||
	    !try_module_get(sst->dev->driver->owner)) {
326
		dev_err(dai->dev, "no device available to run\n");
327 328
		ret_val = -ENODEV;
		goto out_ops;
329
	}
330 331 332 333
	stream->ops = sst->ops;
	mutex_unlock(&sst_lock);

	stream->stream_info.str_id = 0;
334

335
	stream->stream_info.arg = substream;
336
	/* allocate memory for SST API set */
337
	runtime->private_data = stream;
L
Lu Guanqun 已提交
338

339 340 341 342
	ret_val = power_up_sst(stream);
	if (ret_val < 0)
		return ret_val;

343 344 345 346 347 348 349 350 351 352
	/* Make sure, that the period size is always even */
	snd_pcm_hw_constraint_step(substream->runtime, 0,
			   SNDRV_PCM_HW_PARAM_PERIODS, 2);

	return snd_pcm_hw_constraint_integer(runtime,
			 SNDRV_PCM_HW_PARAM_PERIODS);
out_ops:
	kfree(stream);
	mutex_unlock(&sst_lock);
	return ret_val;
353 354
}

355 356
static void sst_media_close(struct snd_pcm_substream *substream,
		struct snd_soc_dai *dai)
357 358 359 360 361
{
	struct sst_runtime_stream *stream;
	int ret_val = 0, str_id;

	stream = substream->runtime->private_data;
362 363
	power_down_sst(stream);

364 365
	str_id = stream->stream_info.str_id;
	if (str_id)
366
		ret_val = stream->ops->close(sst->dev, str_id);
367
	module_put(sst->dev->driver->owner);
368
	kfree(stream);
V
Vinod Koul 已提交
369 370
}

371 372
static int sst_media_prepare(struct snd_pcm_substream *substream,
		struct snd_soc_dai *dai)
373 374 375 376 377 378 379
{
	struct sst_runtime_stream *stream;
	int ret_val = 0, str_id;

	stream = substream->runtime->private_data;
	str_id = stream->stream_info.str_id;
	if (stream->stream_info.str_id) {
380
		ret_val = stream->ops->stream_drop(sst->dev, str_id);
381 382 383
		return ret_val;
	}

384
	ret_val = sst_platform_alloc_stream(substream, dai);
385
	if (ret_val <= 0)
386 387 388 389 390 391 392 393 394 395 396
		return ret_val;
	snprintf(substream->pcm->id, sizeof(substream->pcm->id),
			"%d", stream->stream_info.str_id);

	ret_val = sst_platform_init_stream(substream);
	if (ret_val)
		return ret_val;
	substream->runtime->hw.info = SNDRV_PCM_INFO_BLOCK_TRANSFER;
	return ret_val;
}

397 398 399 400 401 402 403 404 405 406 407 408 409 410 411
static int sst_media_hw_params(struct snd_pcm_substream *substream,
				struct snd_pcm_hw_params *params,
				struct snd_soc_dai *dai)
{
	snd_pcm_lib_malloc_pages(substream, params_buffer_bytes(params));
	memset(substream->runtime->dma_area, 0, params_buffer_bytes(params));
	return 0;
}

static int sst_media_hw_free(struct snd_pcm_substream *substream,
		struct snd_soc_dai *dai)
{
	return snd_pcm_lib_free_pages(substream);
}

412 413 414 415 416 417 418
static int sst_enable_ssp(struct snd_pcm_substream *substream,
			struct snd_soc_dai *dai)
{
	int ret = 0;

	if (!dai->active) {
		ret = sst_handle_vb_timer(dai, true);
419
		sst_fill_ssp_defaults(dai);
420 421 422 423
	}
	return ret;
}

424 425 426 427 428 429 430 431 432 433 434
static int sst_be_hw_params(struct snd_pcm_substream *substream,
				struct snd_pcm_hw_params *params,
				struct snd_soc_dai *dai)
{
	int ret = 0;

	if (dai->active == 1)
		ret = send_ssp_cmd(dai, dai->name, 1);
	return ret;
}

435 436 437 438 439 440 441 442 443 444 445 446 447 448
static int sst_set_format(struct snd_soc_dai *dai, unsigned int fmt)
{
	int ret = 0;

	if (!dai->active)
		return 0;

	ret = sst_fill_ssp_config(dai, fmt);
	if (ret < 0)
		dev_err(dai->dev, "sst_set_format failed..\n");

	return ret;
}

449 450 451 452 453 454 455 456 457 458 459 460 461 462 463
static int sst_platform_set_ssp_slot(struct snd_soc_dai *dai,
			unsigned int tx_mask, unsigned int rx_mask,
			int slots, int slot_width) {
	int ret = 0;

	if (!dai->active)
		return ret;

	ret = sst_fill_ssp_slot(dai, tx_mask, rx_mask, slots, slot_width);
	if (ret < 0)
		dev_err(dai->dev, "sst_fill_ssp_slot failed..%d\n", ret);

	return ret;
}

464 465 466 467 468 469 470 471 472
static void sst_disable_ssp(struct snd_pcm_substream *substream,
			struct snd_soc_dai *dai)
{
	if (!dai->active) {
		send_ssp_cmd(dai, dai->name, 0);
		sst_handle_vb_timer(dai, false);
	}
}

473 474 475 476 477 478
static struct snd_soc_dai_ops sst_media_dai_ops = {
	.startup = sst_media_open,
	.shutdown = sst_media_close,
	.prepare = sst_media_prepare,
	.hw_params = sst_media_hw_params,
	.hw_free = sst_media_hw_free,
479 480 481 482 483 484 485 486 487
	.mute_stream = sst_media_digital_mute,
};

static struct snd_soc_dai_ops sst_compr_dai_ops = {
	.mute_stream = sst_media_digital_mute,
};

static struct snd_soc_dai_ops sst_be_dai_ops = {
	.startup = sst_enable_ssp,
488
	.hw_params = sst_be_hw_params,
489
	.set_fmt = sst_set_format,
490
	.set_tdm_slot = sst_platform_set_ssp_slot,
491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579
	.shutdown = sst_disable_ssp,
};

static struct snd_soc_dai_driver sst_platform_dai[] = {
{
	.name = "media-cpu-dai",
	.ops = &sst_media_dai_ops,
	.playback = {
		.stream_name = "Headset Playback",
		.channels_min = SST_STEREO,
		.channels_max = SST_STEREO,
		.rates = SNDRV_PCM_RATE_44100|SNDRV_PCM_RATE_48000,
		.formats = SNDRV_PCM_FMTBIT_S16_LE,
	},
	.capture = {
		.stream_name = "Headset Capture",
		.channels_min = 1,
		.channels_max = 2,
		.rates = SNDRV_PCM_RATE_44100|SNDRV_PCM_RATE_48000,
		.formats = SNDRV_PCM_FMTBIT_S16_LE,
	},
},
{
	.name = "compress-cpu-dai",
	.compress_dai = 1,
	.ops = &sst_compr_dai_ops,
	.playback = {
		.stream_name = "Compress Playback",
		.channels_min = SST_STEREO,
		.channels_max = SST_STEREO,
		.rates = SNDRV_PCM_RATE_48000,
		.formats = SNDRV_PCM_FMTBIT_S16_LE,
	},
},
/* BE CPU  Dais */
{
	.name = "ssp0-port",
	.ops = &sst_be_dai_ops,
	.playback = {
		.stream_name = "ssp0 Tx",
		.channels_min = SST_STEREO,
		.channels_max = SST_STEREO,
		.rates = SNDRV_PCM_RATE_48000,
		.formats = SNDRV_PCM_FMTBIT_S16_LE,
	},
	.capture = {
		.stream_name = "ssp0 Rx",
		.channels_min = SST_STEREO,
		.channels_max = SST_STEREO,
		.rates = SNDRV_PCM_RATE_48000,
		.formats = SNDRV_PCM_FMTBIT_S16_LE,
	},
},
{
	.name = "ssp1-port",
	.ops = &sst_be_dai_ops,
	.playback = {
		.stream_name = "ssp1 Tx",
		.channels_min = SST_STEREO,
		.channels_max = SST_STEREO,
		.rates = SNDRV_PCM_RATE_8000|SNDRV_PCM_RATE_16000|SNDRV_PCM_RATE_48000,
		.formats = SNDRV_PCM_FMTBIT_S16_LE,
	},
	.capture = {
		.stream_name = "ssp1 Rx",
		.channels_min = SST_STEREO,
		.channels_max = SST_STEREO,
		.rates = SNDRV_PCM_RATE_8000|SNDRV_PCM_RATE_16000|SNDRV_PCM_RATE_48000,
		.formats = SNDRV_PCM_FMTBIT_S16_LE,
	},
},
{
	.name = "ssp2-port",
	.ops = &sst_be_dai_ops,
	.playback = {
		.stream_name = "ssp2 Tx",
		.channels_min = SST_STEREO,
		.channels_max = SST_STEREO,
		.rates = SNDRV_PCM_RATE_48000,
		.formats = SNDRV_PCM_FMTBIT_S16_LE,
	},
	.capture = {
		.stream_name = "ssp2 Rx",
		.channels_min = SST_STEREO,
		.channels_max = SST_STEREO,
		.rates = SNDRV_PCM_RATE_48000,
		.formats = SNDRV_PCM_FMTBIT_S16_LE,
	},
},
580 581 582 583 584 585 586 587 588 589 590 591 592 593
};

static int sst_platform_open(struct snd_pcm_substream *substream)
{
	struct snd_pcm_runtime *runtime;

	if (substream->pcm->internal)
		return 0;

	runtime = substream->runtime;
	runtime->hw = sst_platform_pcm_hw;
	return 0;
}

594
static int sst_platform_pcm_trigger(struct snd_pcm_substream *substream,
595 596 597 598
					int cmd)
{
	int ret_val = 0, str_id;
	struct sst_runtime_stream *stream;
599
	int status;
600
	struct snd_soc_pcm_runtime *rtd = substream->private_data;
601

602
	dev_dbg(rtd->dev, "sst_platform_pcm_trigger called\n");
603 604
	if (substream->pcm->internal)
		return 0;
605 606 607 608
	stream = substream->runtime->private_data;
	str_id = stream->stream_info.str_id;
	switch (cmd) {
	case SNDRV_PCM_TRIGGER_START:
609
		dev_dbg(rtd->dev, "sst: Trigger Start\n");
610
		status = SST_PLATFORM_RUNNING;
611
		stream->stream_info.arg = substream;
612
		ret_val = stream->ops->stream_start(sst->dev, str_id);
613 614
		break;
	case SNDRV_PCM_TRIGGER_STOP:
615
		dev_dbg(rtd->dev, "sst: in stop\n");
616
		status = SST_PLATFORM_DROPPED;
617
		ret_val = stream->ops->stream_drop(sst->dev, str_id);
618 619
		break;
	case SNDRV_PCM_TRIGGER_PAUSE_PUSH:
620
	case SNDRV_PCM_TRIGGER_SUSPEND:
621
		dev_dbg(rtd->dev, "sst: in pause\n");
622
		status = SST_PLATFORM_PAUSED;
623
		ret_val = stream->ops->stream_pause(sst->dev, str_id);
624 625
		break;
	case SNDRV_PCM_TRIGGER_PAUSE_RELEASE:
626
	case SNDRV_PCM_TRIGGER_RESUME:
627
		dev_dbg(rtd->dev, "sst: in pause release\n");
628
		status = SST_PLATFORM_RUNNING;
629
		ret_val = stream->ops->stream_pause_release(sst->dev, str_id);
630 631
		break;
	default:
632
		return -EINVAL;
633
	}
634

635 636 637
	if (!ret_val)
		sst_set_stream_status(stream, status);

638 639 640 641
	return ret_val;
}


642
static snd_pcm_uframes_t sst_platform_pcm_pointer
643 644 645
			(struct snd_pcm_substream *substream)
{
	struct sst_runtime_stream *stream;
646
	int ret_val, status;
647
	struct pcm_stream_info *str_info;
648
	struct snd_soc_pcm_runtime *rtd = substream->private_data;
649 650

	stream = substream->runtime->private_data;
651 652
	status = sst_get_stream_status(stream);
	if (status == SST_PLATFORM_INIT)
653 654
		return 0;
	str_info = &stream->stream_info;
655
	ret_val = stream->ops->stream_read_tstamp(sst->dev, str_info);
656
	if (ret_val) {
657
		dev_err(rtd->dev, "sst: error code = %d\n", ret_val);
658 659
		return ret_val;
	}
660
	substream->runtime->delay = str_info->pcm_delay;
661
	return str_info->buffer_ptr;
662 663
}

664
static struct snd_pcm_ops sst_platform_ops = {
665
	.open = sst_platform_open,
666
	.ioctl = snd_pcm_lib_ioctl,
667 668
	.trigger = sst_platform_pcm_trigger,
	.pointer = sst_platform_pcm_pointer,
669 670
};

671
static int sst_pcm_new(struct snd_soc_pcm_runtime *rtd)
672
{
673
	struct snd_soc_dai *dai = rtd->cpu_dai;
674
	struct snd_pcm *pcm = rtd->pcm;
675 676
	int retval = 0;

677 678
	if (dai->driver->playback.channels_min ||
			dai->driver->capture.channels_min) {
679 680
		retval =  snd_pcm_lib_preallocate_pages_for_all(pcm,
			SNDRV_DMA_TYPE_CONTINUOUS,
681
			snd_dma_continuous_data(GFP_DMA),
682 683
			SST_MIN_BUFFER, SST_MAX_BUFFER);
		if (retval) {
684
			dev_err(rtd->dev, "dma buffer allocationf fail\n");
685 686 687 688 689
			return retval;
		}
	}
	return retval;
}
690

V
Vinod Koul 已提交
691 692
static int sst_soc_probe(struct snd_soc_platform *platform)
{
693 694 695
	struct sst_data *drv = dev_get_drvdata(platform->dev);

	drv->soc_card = platform->component.card;
V
Vinod Koul 已提交
696 697 698 699 700
	return sst_dsp_init_v2_dpcm(platform);
}

static struct snd_soc_platform_driver sst_soc_platform_drv  = {
	.probe		= sst_soc_probe,
701
	.ops		= &sst_platform_ops,
702
	.compr_ops	= &sst_platform_compr_ops,
703 704 705
	.pcm_new	= sst_pcm_new,
};

706 707 708 709 710
static const struct snd_soc_component_driver sst_component = {
	.name		= "sst",
};


711 712
static int sst_platform_probe(struct platform_device *pdev)
{
V
Vinod Koul 已提交
713
	struct sst_data *drv;
714
	int ret;
715
	struct sst_platform_data *pdata;
V
Vinod Koul 已提交
716 717

	drv = devm_kzalloc(&pdev->dev, sizeof(*drv), GFP_KERNEL);
718
	if (drv == NULL) {
V
Vinod Koul 已提交
719 720 721
		return -ENOMEM;
	}

722 723 724 725 726
	pdata = devm_kzalloc(&pdev->dev, sizeof(*pdata), GFP_KERNEL);
	if (pdata == NULL) {
		return -ENOMEM;
	}

V
Vinod Koul 已提交
727 728 729
	pdata->pdev_strm_map = dpcm_strm_map;
	pdata->strm_map_size = ARRAY_SIZE(dpcm_strm_map);
	drv->pdata = pdata;
730
	drv->pdev = pdev;
V
Vinod Koul 已提交
731 732
	mutex_init(&drv->lock);
	dev_set_drvdata(&pdev->dev, drv);
733 734 735

	ret = snd_soc_register_platform(&pdev->dev, &sst_soc_platform_drv);
	if (ret) {
736
		dev_err(&pdev->dev, "registering soc platform failed\n");
737 738
		return ret;
	}
739

740
	ret = snd_soc_register_component(&pdev->dev, &sst_component,
741 742
				sst_platform_dai, ARRAY_SIZE(sst_platform_dai));
	if (ret) {
743
		dev_err(&pdev->dev, "registering cpu dais failed\n");
744 745 746 747 748 749 750 751
		snd_soc_unregister_platform(&pdev->dev);
	}
	return ret;
}

static int sst_platform_remove(struct platform_device *pdev)
{

752
	snd_soc_unregister_component(&pdev->dev);
753
	snd_soc_unregister_platform(&pdev->dev);
754
	dev_dbg(&pdev->dev, "sst_platform_remove success\n");
755 756 757
	return 0;
}

758 759 760 761 762 763 764 765 766 767 768 769 770 771 772 773 774 775 776 777 778 779 780 781 782 783 784 785 786 787 788 789 790 791 792 793 794 795 796 797 798 799 800 801 802 803 804 805 806 807 808 809 810 811
#ifdef CONFIG_PM_SLEEP

static int sst_soc_prepare(struct device *dev)
{
	struct sst_data *drv = dev_get_drvdata(dev);
	int i;

	/* suspend all pcms first */
	snd_soc_suspend(drv->soc_card->dev);
	snd_soc_poweroff(drv->soc_card->dev);

	/* set the SSPs to idle */
	for (i = 0; i < drv->soc_card->num_rtd; i++) {
		struct snd_soc_dai *dai = drv->soc_card->rtd[i].cpu_dai;

		if (dai->active) {
			send_ssp_cmd(dai, dai->name, 0);
			sst_handle_vb_timer(dai, false);
		}
	}

	return 0;
}

static void sst_soc_complete(struct device *dev)
{
	struct sst_data *drv = dev_get_drvdata(dev);
	int i;

	/* restart SSPs */
	for (i = 0; i < drv->soc_card->num_rtd; i++) {
		struct snd_soc_dai *dai = drv->soc_card->rtd[i].cpu_dai;

		if (dai->active) {
			sst_handle_vb_timer(dai, true);
			send_ssp_cmd(dai, dai->name, 1);
		}
	}
	snd_soc_resume(drv->soc_card->dev);
}

#else

#define sst_soc_prepare NULL
#define sst_soc_complete NULL

#endif


static const struct dev_pm_ops sst_platform_pm = {
	.prepare	= sst_soc_prepare,
	.complete	= sst_soc_complete,
};

812 813
static struct platform_driver sst_platform_driver = {
	.driver		= {
814
		.name		= "sst-mfld-platform",
815
		.pm             = &sst_platform_pm,
816 817 818 819 820
	},
	.probe		= sst_platform_probe,
	.remove		= sst_platform_remove,
};

821
module_platform_driver(sst_platform_driver);
822 823 824 825 826

MODULE_DESCRIPTION("ASoC Intel(R) MID Platform driver");
MODULE_AUTHOR("Vinod Koul <vinod.koul@intel.com>");
MODULE_AUTHOR("Harsha Priya <priya.harsha@intel.com>");
MODULE_LICENSE("GPL v2");
827
MODULE_ALIAS("platform:sst-mfld-platform");