pmac.c 37.2 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28
/*
 * PMac DBDMA lowlevel functions
 *
 * Copyright (c) by Takashi Iwai <tiwai@suse.de>
 * code based on dmasound.c.
 *
 *   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.
 *
 *   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.
 *
 *   You should have received a copy of the GNU General Public License
 *   along with this program; if not, write to the Free Software
 *   Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307 USA
 */


#include <asm/io.h>
#include <asm/irq.h>
#include <linux/init.h>
#include <linux/delay.h>
#include <linux/slab.h>
#include <linux/interrupt.h>
29 30
#include <linux/pci.h>
#include <linux/dma-mapping.h>
L
Linus Torvalds 已提交
31 32 33 34
#include <sound/core.h>
#include "pmac.h"
#include <sound/pcm_params.h>
#include <asm/pmac_feature.h>
35
#include <asm/pci-bridge.h>
L
Linus Torvalds 已提交
36 37 38 39 40 41 42 43 44 45 46


/* fixed frequency table for awacs, screamer, burgundy, DACA (44100 max) */
static int awacs_freqs[8] = {
	44100, 29400, 22050, 17640, 14700, 11025, 8820, 7350
};
/* fixed frequency table for tumbler */
static int tumbler_freqs[1] = {
	44100
};

47 48 49 50 51 52 53 54 55 56 57 58

/*
 * we will allocate a single 'emergency' dbdma cmd block to use if the
 * tx status comes up "DEAD".  This happens on some PowerComputing Pmac
 * clones, either owing to a bug in dbdma or some interaction between
 * IDE and sound.  However, this measure would deal with DEAD status if
 * it appeared elsewhere.
 */
static struct pmac_dbdma emergency_dbdma;
static int emergency_in_use;


L
Linus Torvalds 已提交
59 60 61
/*
 * allocate DBDMA command arrays
 */
62
static int snd_pmac_dbdma_alloc(struct snd_pmac *chip, struct pmac_dbdma *rec, int size)
L
Linus Torvalds 已提交
63
{
64 65 66 67
	unsigned int rsize = sizeof(struct dbdma_cmd) * (size + 1);

	rec->space = dma_alloc_coherent(&chip->pdev->dev, rsize,
					&rec->dma_base, GFP_KERNEL);
L
Linus Torvalds 已提交
68 69 70
	if (rec->space == NULL)
		return -ENOMEM;
	rec->size = size;
71
	memset(rec->space, 0, rsize);
L
Linus Torvalds 已提交
72
	rec->cmds = (void __iomem *)DBDMA_ALIGN(rec->space);
73 74
	rec->addr = rec->dma_base + (unsigned long)((char *)rec->cmds - (char *)rec->space);

L
Linus Torvalds 已提交
75 76 77
	return 0;
}

78
static void snd_pmac_dbdma_free(struct snd_pmac *chip, struct pmac_dbdma *rec)
L
Linus Torvalds 已提交
79
{
80
	if (rec->space) {
81 82 83 84
		unsigned int rsize = sizeof(struct dbdma_cmd) * (rec->size + 1);

		dma_free_coherent(&chip->pdev->dev, rsize, rec->space, rec->dma_base);
	}
L
Linus Torvalds 已提交
85 86 87 88 89 90 91 92 93 94 95
}


/*
 * pcm stuff
 */

/*
 * look up frequency table
 */

96
unsigned int snd_pmac_rate_index(struct snd_pmac *chip, struct pmac_stream *rec, unsigned int rate)
L
Linus Torvalds 已提交
97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124
{
	int i, ok, found;

	ok = rec->cur_freqs;
	if (rate > chip->freq_table[0])
		return 0;
	found = 0;
	for (i = 0; i < chip->num_freqs; i++, ok >>= 1) {
		if (! (ok & 1)) continue;
		found = i;
		if (rate >= chip->freq_table[i])
			break;
	}
	return found;
}

/*
 * check whether another stream is active
 */
static inline int another_stream(int stream)
{
	return (stream == SNDRV_PCM_STREAM_PLAYBACK) ?
		SNDRV_PCM_STREAM_CAPTURE : SNDRV_PCM_STREAM_PLAYBACK;
}

/*
 * allocate buffers
 */
125 126
static int snd_pmac_pcm_hw_params(struct snd_pcm_substream *subs,
				  struct snd_pcm_hw_params *hw_params)
L
Linus Torvalds 已提交
127 128 129 130 131 132 133
{
	return snd_pcm_lib_malloc_pages(subs, params_buffer_bytes(hw_params));
}

/*
 * release buffers
 */
134
static int snd_pmac_pcm_hw_free(struct snd_pcm_substream *subs)
L
Linus Torvalds 已提交
135 136 137 138 139 140 141 142
{
	snd_pcm_lib_free_pages(subs);
	return 0;
}

/*
 * get a stream of the opposite direction
 */
143
static struct pmac_stream *snd_pmac_get_stream(struct snd_pmac *chip, int stream)
L
Linus Torvalds 已提交
144 145 146 147 148 149 150 151 152 153 154 155 156 157 158
{
	switch (stream) {
	case SNDRV_PCM_STREAM_PLAYBACK:
		return &chip->playback;
	case SNDRV_PCM_STREAM_CAPTURE:
		return &chip->capture;
	default:
		snd_BUG();
		return NULL;
	}
}

/*
 * wait while run status is on
 */
159
static inline void
160
snd_pmac_wait_ack(struct pmac_stream *rec)
L
Linus Torvalds 已提交
161 162 163 164 165 166 167 168 169 170
{
	int timeout = 50000;
	while ((in_le32(&rec->dma->status) & RUN) && timeout-- > 0)
		udelay(1);
}

/*
 * set the format and rate to the chip.
 * call the lowlevel function if defined (e.g. for AWACS).
 */
171
static void snd_pmac_pcm_set_format(struct snd_pmac *chip)
L
Linus Torvalds 已提交
172 173 174 175 176 177 178 179 180 181 182
{
	/* set up frequency and format */
	out_le32(&chip->awacs->control, chip->control_mask | (chip->rate_index << 8));
	out_le32(&chip->awacs->byteswap, chip->format == SNDRV_PCM_FORMAT_S16_LE ? 1 : 0);
	if (chip->set_format)
		chip->set_format(chip);
}

/*
 * stop the DMA transfer
 */
183
static inline void snd_pmac_dma_stop(struct pmac_stream *rec)
L
Linus Torvalds 已提交
184 185 186 187 188 189 190 191
{
	out_le32(&rec->dma->control, (RUN|WAKE|FLUSH|PAUSE) << 16);
	snd_pmac_wait_ack(rec);
}

/*
 * set the command pointer address
 */
192
static inline void snd_pmac_dma_set_command(struct pmac_stream *rec, struct pmac_dbdma *cmd)
L
Linus Torvalds 已提交
193 194 195 196 197 198 199
{
	out_le32(&rec->dma->cmdptr, cmd->addr);
}

/*
 * start the DMA
 */
200
static inline void snd_pmac_dma_run(struct pmac_stream *rec, int status)
L
Linus Torvalds 已提交
201 202 203 204 205 206 207 208
{
	out_le32(&rec->dma->control, status | (status << 16));
}


/*
 * prepare playback/capture stream
 */
209
static int snd_pmac_pcm_prepare(struct snd_pmac *chip, struct pmac_stream *rec, struct snd_pcm_substream *subs)
L
Linus Torvalds 已提交
210 211 212
{
	int i;
	volatile struct dbdma_cmd __iomem *cp;
213
	struct snd_pcm_runtime *runtime = subs->runtime;
L
Linus Torvalds 已提交
214 215
	int rate_index;
	long offset;
216
	struct pmac_stream *astr;
217

L
Linus Torvalds 已提交
218 219 220 221 222 223 224 225
	rec->dma_size = snd_pcm_lib_buffer_bytes(subs);
	rec->period_size = snd_pcm_lib_period_bytes(subs);
	rec->nperiods = rec->dma_size / rec->period_size;
	rec->cur_period = 0;
	rate_index = snd_pmac_rate_index(chip, rec, runtime->rate);

	/* set up constraints */
	astr = snd_pmac_get_stream(chip, another_stream(rec->stream));
226 227
	if (! astr)
		return -EINVAL;
L
Linus Torvalds 已提交
228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249
	astr->cur_freqs = 1 << rate_index;
	astr->cur_formats = 1 << runtime->format;
	chip->rate_index = rate_index;
	chip->format = runtime->format;

	/* We really want to execute a DMA stop command, after the AWACS
	 * is initialized.
	 * For reasons I don't understand, it stops the hissing noise
	 * common to many PowerBook G3 systems and random noise otherwise
	 * captured on iBook2's about every third time. -ReneR
	 */
	spin_lock_irq(&chip->reg_lock);
	snd_pmac_dma_stop(rec);
	st_le16(&chip->extra_dma.cmds->command, DBDMA_STOP);
	snd_pmac_dma_set_command(rec, &chip->extra_dma);
	snd_pmac_dma_run(rec, RUN);
	spin_unlock_irq(&chip->reg_lock);
	mdelay(5);
	spin_lock_irq(&chip->reg_lock);
	/* continuous DMA memory type doesn't provide the physical address,
	 * so we need to resolve the address here...
	 */
250
	offset = runtime->dma_addr;
L
Linus Torvalds 已提交
251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272
	for (i = 0, cp = rec->cmd.cmds; i < rec->nperiods; i++, cp++) {
		st_le32(&cp->phy_addr, offset);
		st_le16(&cp->req_count, rec->period_size);
		/*st_le16(&cp->res_count, 0);*/
		st_le16(&cp->xfer_status, 0);
		offset += rec->period_size;
	}
	/* make loop */
	st_le16(&cp->command, DBDMA_NOP + BR_ALWAYS);
	st_le32(&cp->cmd_dep, rec->cmd.addr);

	snd_pmac_dma_stop(rec);
	snd_pmac_dma_set_command(rec, &rec->cmd);
	spin_unlock_irq(&chip->reg_lock);

	return 0;
}


/*
 * PCM trigger/stop
 */
273 274
static int snd_pmac_pcm_trigger(struct snd_pmac *chip, struct pmac_stream *rec,
				struct snd_pcm_substream *subs, int cmd)
L
Linus Torvalds 已提交
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 317 318 319
{
	volatile struct dbdma_cmd __iomem *cp;
	int i, command;

	switch (cmd) {
	case SNDRV_PCM_TRIGGER_START:
	case SNDRV_PCM_TRIGGER_RESUME:
		if (rec->running)
			return -EBUSY;
		command = (subs->stream == SNDRV_PCM_STREAM_PLAYBACK ?
			   OUTPUT_MORE : INPUT_MORE) + INTR_ALWAYS;
		spin_lock(&chip->reg_lock);
		snd_pmac_beep_stop(chip);
		snd_pmac_pcm_set_format(chip);
		for (i = 0, cp = rec->cmd.cmds; i < rec->nperiods; i++, cp++)
			out_le16(&cp->command, command);
		snd_pmac_dma_set_command(rec, &rec->cmd);
		(void)in_le32(&rec->dma->status);
		snd_pmac_dma_run(rec, RUN|WAKE);
		rec->running = 1;
		spin_unlock(&chip->reg_lock);
		break;

	case SNDRV_PCM_TRIGGER_STOP:
	case SNDRV_PCM_TRIGGER_SUSPEND:
		spin_lock(&chip->reg_lock);
		rec->running = 0;
		/*printk("stopped!!\n");*/
		snd_pmac_dma_stop(rec);
		for (i = 0, cp = rec->cmd.cmds; i < rec->nperiods; i++, cp++)
			out_le16(&cp->command, DBDMA_STOP);
		spin_unlock(&chip->reg_lock);
		break;

	default:
		return -EINVAL;
	}

	return 0;
}

/*
 * return the current pointer
 */
inline
320 321 322
static snd_pcm_uframes_t snd_pmac_pcm_pointer(struct snd_pmac *chip,
					      struct pmac_stream *rec,
					      struct snd_pcm_substream *subs)
L
Linus Torvalds 已提交
323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344
{
	int count = 0;

#if 1 /* hmm.. how can we get the current dma pointer?? */
	int stat;
	volatile struct dbdma_cmd __iomem *cp = &rec->cmd.cmds[rec->cur_period];
	stat = ld_le16(&cp->xfer_status);
	if (stat & (ACTIVE|DEAD)) {
		count = in_le16(&cp->res_count);
		if (count)
			count = rec->period_size - count;
	}
#endif
	count += rec->cur_period * rec->period_size;
	/*printk("pointer=%d\n", count);*/
	return bytes_to_frames(subs->runtime, count);
}

/*
 * playback
 */

345
static int snd_pmac_playback_prepare(struct snd_pcm_substream *subs)
L
Linus Torvalds 已提交
346
{
347
	struct snd_pmac *chip = snd_pcm_substream_chip(subs);
L
Linus Torvalds 已提交
348 349 350
	return snd_pmac_pcm_prepare(chip, &chip->playback, subs);
}

351
static int snd_pmac_playback_trigger(struct snd_pcm_substream *subs,
L
Linus Torvalds 已提交
352 353
				     int cmd)
{
354
	struct snd_pmac *chip = snd_pcm_substream_chip(subs);
L
Linus Torvalds 已提交
355 356 357
	return snd_pmac_pcm_trigger(chip, &chip->playback, subs, cmd);
}

358
static snd_pcm_uframes_t snd_pmac_playback_pointer(struct snd_pcm_substream *subs)
L
Linus Torvalds 已提交
359
{
360
	struct snd_pmac *chip = snd_pcm_substream_chip(subs);
L
Linus Torvalds 已提交
361 362 363 364 365 366 367 368
	return snd_pmac_pcm_pointer(chip, &chip->playback, subs);
}


/*
 * capture
 */

369
static int snd_pmac_capture_prepare(struct snd_pcm_substream *subs)
L
Linus Torvalds 已提交
370
{
371
	struct snd_pmac *chip = snd_pcm_substream_chip(subs);
L
Linus Torvalds 已提交
372 373 374
	return snd_pmac_pcm_prepare(chip, &chip->capture, subs);
}

375
static int snd_pmac_capture_trigger(struct snd_pcm_substream *subs,
L
Linus Torvalds 已提交
376 377
				    int cmd)
{
378
	struct snd_pmac *chip = snd_pcm_substream_chip(subs);
L
Linus Torvalds 已提交
379 380 381
	return snd_pmac_pcm_trigger(chip, &chip->capture, subs, cmd);
}

382
static snd_pcm_uframes_t snd_pmac_capture_pointer(struct snd_pcm_substream *subs)
L
Linus Torvalds 已提交
383
{
384
	struct snd_pmac *chip = snd_pcm_substream_chip(subs);
L
Linus Torvalds 已提交
385 386 387 388
	return snd_pmac_pcm_pointer(chip, &chip->capture, subs);
}


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 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457
/*
 * Handle DEAD DMA transfers:
 * if the TX status comes up "DEAD" - reported on some Power Computing machines
 * we need to re-start the dbdma - but from a different physical start address
 * and with a different transfer length.  It would get very messy to do this
 * with the normal dbdma_cmd blocks - we would have to re-write the buffer start
 * addresses each time.  So, we will keep a single dbdma_cmd block which can be
 * fiddled with.
 * When DEAD status is first reported the content of the faulted dbdma block is
 * copied into the emergency buffer and we note that the buffer is in use.
 * we then bump the start physical address by the amount that was successfully
 * output before it died.
 * On any subsequent DEAD result we just do the bump-ups (we know that we are
 * already using the emergency dbdma_cmd).
 * CHECK: this just tries to "do it".  It is possible that we should abandon
 * xfers when the number of residual bytes gets below a certain value - I can
 * see that this might cause a loop-forever if a too small transfer causes
 * DEAD status.  However this is a TODO for now - we'll see what gets reported.
 * When we get a successful transfer result with the emergency buffer we just
 * pretend that it completed using the original dmdma_cmd and carry on.  The
 * 'next_cmd' field will already point back to the original loop of blocks.
 */
static inline void snd_pmac_pcm_dead_xfer(struct pmac_stream *rec,
					  volatile struct dbdma_cmd __iomem *cp)
{
	unsigned short req, res ;
	unsigned int phy ;

	/* printk(KERN_WARNING "snd-powermac: DMA died - patching it up!\n"); */

	/* to clear DEAD status we must first clear RUN
	   set it to quiescent to be on the safe side */
	(void)in_le32(&rec->dma->status);
	out_le32(&rec->dma->control, (RUN|PAUSE|FLUSH|WAKE) << 16);

	if (!emergency_in_use) { /* new problem */
		memcpy((void *)emergency_dbdma.cmds, (void *)cp,
		       sizeof(struct dbdma_cmd));
		emergency_in_use = 1;
		st_le16(&cp->xfer_status, 0);
		st_le16(&cp->req_count, rec->period_size);
		cp = emergency_dbdma.cmds;
	}

	/* now bump the values to reflect the amount
	   we haven't yet shifted */
	req = ld_le16(&cp->req_count);
	res = ld_le16(&cp->res_count);
	phy = ld_le32(&cp->phy_addr);
	phy += (req - res);
	st_le16(&cp->req_count, res);
	st_le16(&cp->res_count, 0);
	st_le16(&cp->xfer_status, 0);
	st_le32(&cp->phy_addr, phy);

	st_le32(&cp->cmd_dep, rec->cmd.addr
		+ sizeof(struct dbdma_cmd)*((rec->cur_period+1)%rec->nperiods));

	st_le16(&cp->command, OUTPUT_MORE | BR_ALWAYS | INTR_ALWAYS);

	/* point at our patched up command block */
	out_le32(&rec->dma->cmdptr, emergency_dbdma.addr);

	/* we must re-start the controller */
	(void)in_le32(&rec->dma->status);
	/* should complete clearing the DEAD status */
	out_le32(&rec->dma->control, ((RUN|WAKE) << 16) + (RUN|WAKE));
}

L
Linus Torvalds 已提交
458 459 460
/*
 * update playback/capture pointer from interrupts
 */
461
static void snd_pmac_pcm_update(struct snd_pmac *chip, struct pmac_stream *rec)
L
Linus Torvalds 已提交
462 463 464 465 466 467 468 469
{
	volatile struct dbdma_cmd __iomem *cp;
	int c;
	int stat;

	spin_lock(&chip->reg_lock);
	if (rec->running) {
		for (c = 0; c < rec->nperiods; c++) { /* at most all fragments */
470 471 472 473 474 475

			if (emergency_in_use)   /* already using DEAD xfer? */
				cp = emergency_dbdma.cmds;
			else
				cp = &rec->cmd.cmds[rec->cur_period];

L
Linus Torvalds 已提交
476
			stat = ld_le16(&cp->xfer_status);
477 478 479 480 481 482 483 484 485

			if (stat & DEAD) {
				snd_pmac_pcm_dead_xfer(rec, cp);
				break; /* this block is still going */
			}

			if (emergency_in_use)
				emergency_in_use = 0 ; /* done that */

L
Linus Torvalds 已提交
486 487
			if (! (stat & ACTIVE))
				break;
488

L
Linus Torvalds 已提交
489 490 491 492 493 494 495
			/*printk("update frag %d\n", rec->cur_period);*/
			st_le16(&cp->xfer_status, 0);
			st_le16(&cp->req_count, rec->period_size);
			/*st_le16(&cp->res_count, 0);*/
			rec->cur_period++;
			if (rec->cur_period >= rec->nperiods) {
				rec->cur_period = 0;
496 497
			}

L
Linus Torvalds 已提交
498 499 500 501 502 503 504 505 506 507 508 509 510
			spin_unlock(&chip->reg_lock);
			snd_pcm_period_elapsed(rec->substream);
			spin_lock(&chip->reg_lock);
		}
	}
	spin_unlock(&chip->reg_lock);
}


/*
 * hw info
 */

511
static struct snd_pcm_hardware snd_pmac_playback =
L
Linus Torvalds 已提交
512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529
{
	.info =			(SNDRV_PCM_INFO_INTERLEAVED |
				 SNDRV_PCM_INFO_MMAP |
				 SNDRV_PCM_INFO_MMAP_VALID |
				 SNDRV_PCM_INFO_RESUME),
	.formats =		SNDRV_PCM_FMTBIT_S16_BE | SNDRV_PCM_FMTBIT_S16_LE,
	.rates =		SNDRV_PCM_RATE_8000_44100,
	.rate_min =		7350,
	.rate_max =		44100,
	.channels_min =		2,
	.channels_max =		2,
	.buffer_bytes_max =	131072,
	.period_bytes_min =	256,
	.period_bytes_max =	16384,
	.periods_min =		3,
	.periods_max =		PMAC_MAX_FRAGS,
};

530
static struct snd_pcm_hardware snd_pmac_capture =
L
Linus Torvalds 已提交
531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550
{
	.info =			(SNDRV_PCM_INFO_INTERLEAVED |
				 SNDRV_PCM_INFO_MMAP |
				 SNDRV_PCM_INFO_MMAP_VALID |
				 SNDRV_PCM_INFO_RESUME),
	.formats =		SNDRV_PCM_FMTBIT_S16_BE | SNDRV_PCM_FMTBIT_S16_LE,
	.rates =		SNDRV_PCM_RATE_8000_44100,
	.rate_min =		7350,
	.rate_max =		44100,
	.channels_min =		2,
	.channels_max =		2,
	.buffer_bytes_max =	131072,
	.period_bytes_min =	256,
	.period_bytes_max =	16384,
	.periods_min =		3,
	.periods_max =		PMAC_MAX_FRAGS,
};


#if 0 // NYI
551 552
static int snd_pmac_hw_rule_rate(struct snd_pcm_hw_params *params,
				 struct snd_pcm_hw_rule *rule)
L
Linus Torvalds 已提交
553
{
554 555
	struct snd_pmac *chip = rule->private;
	struct pmac_stream *rec = snd_pmac_get_stream(chip, rule->deps[0]);
L
Linus Torvalds 已提交
556 557
	int i, freq_table[8], num_freqs;

558 559
	if (! rec)
		return -EINVAL;
L
Linus Torvalds 已提交
560 561 562 563 564 565 566 567 568 569
	num_freqs = 0;
	for (i = chip->num_freqs - 1; i >= 0; i--) {
		if (rec->cur_freqs & (1 << i))
			freq_table[num_freqs++] = chip->freq_table[i];
	}

	return snd_interval_list(hw_param_interval(params, rule->var),
				 num_freqs, freq_table, 0);
}

570 571
static int snd_pmac_hw_rule_format(struct snd_pcm_hw_params *params,
				   struct snd_pcm_hw_rule *rule)
L
Linus Torvalds 已提交
572
{
573 574
	struct snd_pmac *chip = rule->private;
	struct pmac_stream *rec = snd_pmac_get_stream(chip, rule->deps[0]);
L
Linus Torvalds 已提交
575

576 577
	if (! rec)
		return -EINVAL;
L
Linus Torvalds 已提交
578 579 580 581 582
	return snd_mask_refine_set(hw_param_mask(params, SNDRV_PCM_HW_PARAM_FORMAT),
				   rec->cur_formats);
}
#endif // NYI

583 584
static int snd_pmac_pcm_open(struct snd_pmac *chip, struct pmac_stream *rec,
			     struct snd_pcm_substream *subs)
L
Linus Torvalds 已提交
585
{
586
	struct snd_pcm_runtime *runtime = subs->runtime;
587
	int i;
L
Linus Torvalds 已提交
588 589 590

	/* look up frequency table and fill bit mask */
	runtime->hw.rates = 0;
591 592 593 594
	for (i = 0; i < chip->num_freqs; i++)
		if (chip->freqs_ok & (1 << i))
			runtime->hw.rates |=
				snd_pcm_rate_to_rate_bit(chip->freq_table[i]);
L
Linus Torvalds 已提交
595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631

	/* check for minimum and maximum rates */
	for (i = 0; i < chip->num_freqs; i++) {
		if (chip->freqs_ok & (1 << i)) {
			runtime->hw.rate_max = chip->freq_table[i];
			break;
		}
	}
	for (i = chip->num_freqs - 1; i >= 0; i--) {
		if (chip->freqs_ok & (1 << i)) {
			runtime->hw.rate_min = chip->freq_table[i];
			break;
		}
	}
	runtime->hw.formats = chip->formats_ok;
	if (chip->can_capture) {
		if (! chip->can_duplex)
			runtime->hw.info |= SNDRV_PCM_INFO_HALF_DUPLEX;
		runtime->hw.info |= SNDRV_PCM_INFO_JOINT_DUPLEX;
	}
	runtime->private_data = rec;
	rec->substream = subs;

#if 0 /* FIXME: still under development.. */
	snd_pcm_hw_rule_add(runtime, 0, SNDRV_PCM_HW_PARAM_RATE,
			    snd_pmac_hw_rule_rate, chip, rec->stream, -1);
	snd_pcm_hw_rule_add(runtime, 0, SNDRV_PCM_HW_PARAM_FORMAT,
			    snd_pmac_hw_rule_format, chip, rec->stream, -1);
#endif

	runtime->hw.periods_max = rec->cmd.size - 1;

	/* constraints to fix choppy sound */
	snd_pcm_hw_constraint_integer(runtime, SNDRV_PCM_HW_PARAM_PERIODS);
	return 0;
}

632 633
static int snd_pmac_pcm_close(struct snd_pmac *chip, struct pmac_stream *rec,
			      struct snd_pcm_substream *subs)
L
Linus Torvalds 已提交
634
{
635
	struct pmac_stream *astr;
L
Linus Torvalds 已提交
636 637 638 639

	snd_pmac_dma_stop(rec);

	astr = snd_pmac_get_stream(chip, another_stream(rec->stream));
640 641
	if (! astr)
		return -EINVAL;
L
Linus Torvalds 已提交
642 643 644 645

	/* reset constraints */
	astr->cur_freqs = chip->freqs_ok;
	astr->cur_formats = chip->formats_ok;
646

L
Linus Torvalds 已提交
647 648 649
	return 0;
}

650
static int snd_pmac_playback_open(struct snd_pcm_substream *subs)
L
Linus Torvalds 已提交
651
{
652
	struct snd_pmac *chip = snd_pcm_substream_chip(subs);
L
Linus Torvalds 已提交
653 654 655 656 657

	subs->runtime->hw = snd_pmac_playback;
	return snd_pmac_pcm_open(chip, &chip->playback, subs);
}

658
static int snd_pmac_capture_open(struct snd_pcm_substream *subs)
L
Linus Torvalds 已提交
659
{
660
	struct snd_pmac *chip = snd_pcm_substream_chip(subs);
L
Linus Torvalds 已提交
661 662 663 664 665

	subs->runtime->hw = snd_pmac_capture;
	return snd_pmac_pcm_open(chip, &chip->capture, subs);
}

666
static int snd_pmac_playback_close(struct snd_pcm_substream *subs)
L
Linus Torvalds 已提交
667
{
668
	struct snd_pmac *chip = snd_pcm_substream_chip(subs);
L
Linus Torvalds 已提交
669 670 671 672

	return snd_pmac_pcm_close(chip, &chip->playback, subs);
}

673
static int snd_pmac_capture_close(struct snd_pcm_substream *subs)
L
Linus Torvalds 已提交
674
{
675
	struct snd_pmac *chip = snd_pcm_substream_chip(subs);
L
Linus Torvalds 已提交
676 677 678 679 680 681 682

	return snd_pmac_pcm_close(chip, &chip->capture, subs);
}

/*
 */

683
static struct snd_pcm_ops snd_pmac_playback_ops = {
L
Linus Torvalds 已提交
684 685 686 687 688 689 690 691 692 693
	.open =		snd_pmac_playback_open,
	.close =	snd_pmac_playback_close,
	.ioctl =	snd_pcm_lib_ioctl,
	.hw_params =	snd_pmac_pcm_hw_params,
	.hw_free =	snd_pmac_pcm_hw_free,
	.prepare =	snd_pmac_playback_prepare,
	.trigger =	snd_pmac_playback_trigger,
	.pointer =	snd_pmac_playback_pointer,
};

694
static struct snd_pcm_ops snd_pmac_capture_ops = {
L
Linus Torvalds 已提交
695 696 697 698 699 700 701 702 703 704
	.open =		snd_pmac_capture_open,
	.close =	snd_pmac_capture_close,
	.ioctl =	snd_pcm_lib_ioctl,
	.hw_params =	snd_pmac_pcm_hw_params,
	.hw_free =	snd_pmac_pcm_hw_free,
	.prepare =	snd_pmac_capture_prepare,
	.trigger =	snd_pmac_capture_trigger,
	.pointer =	snd_pmac_capture_pointer,
};

705
int __init snd_pmac_pcm_new(struct snd_pmac *chip)
L
Linus Torvalds 已提交
706
{
707
	struct snd_pcm *pcm;
L
Linus Torvalds 已提交
708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 725 726 727 728 729 730 731 732 733 734 735
	int err;
	int num_captures = 1;

	if (! chip->can_capture)
		num_captures = 0;
	err = snd_pcm_new(chip->card, chip->card->driver, 0, 1, num_captures, &pcm);
	if (err < 0)
		return err;

	snd_pcm_set_ops(pcm, SNDRV_PCM_STREAM_PLAYBACK, &snd_pmac_playback_ops);
	if (chip->can_capture)
		snd_pcm_set_ops(pcm, SNDRV_PCM_STREAM_CAPTURE, &snd_pmac_capture_ops);

	pcm->private_data = chip;
	pcm->info_flags = SNDRV_PCM_INFO_JOINT_DUPLEX;
	strcpy(pcm->name, chip->card->shortname);
	chip->pcm = pcm;

	chip->formats_ok = SNDRV_PCM_FMTBIT_S16_BE;
	if (chip->can_byte_swap)
		chip->formats_ok |= SNDRV_PCM_FMTBIT_S16_LE;

	chip->playback.cur_formats = chip->formats_ok;
	chip->capture.cur_formats = chip->formats_ok;
	chip->playback.cur_freqs = chip->freqs_ok;
	chip->capture.cur_freqs = chip->freqs_ok;

	/* preallocate 64k buffer */
736 737
	snd_pcm_lib_preallocate_pages_for_all(pcm, SNDRV_DMA_TYPE_DEV,
					      &chip->pdev->dev,
L
Linus Torvalds 已提交
738 739 740 741 742 743
					      64 * 1024, 64 * 1024);

	return 0;
}


744
static void snd_pmac_dbdma_reset(struct snd_pmac *chip)
L
Linus Torvalds 已提交
745 746 747 748 749 750 751 752 753 754 755
{
	out_le32(&chip->playback.dma->control, (RUN|PAUSE|FLUSH|WAKE|DEAD) << 16);
	snd_pmac_wait_ack(&chip->playback);
	out_le32(&chip->capture.dma->control, (RUN|PAUSE|FLUSH|WAKE|DEAD) << 16);
	snd_pmac_wait_ack(&chip->capture);
}


/*
 * handling beep
 */
756
void snd_pmac_beep_dma_start(struct snd_pmac *chip, int bytes, unsigned long addr, int speed)
L
Linus Torvalds 已提交
757
{
758
	struct pmac_stream *rec = &chip->playback;
L
Linus Torvalds 已提交
759 760 761 762 763 764 765 766 767 768 769 770 771 772 773

	snd_pmac_dma_stop(rec);
	st_le16(&chip->extra_dma.cmds->req_count, bytes);
	st_le16(&chip->extra_dma.cmds->xfer_status, 0);
	st_le32(&chip->extra_dma.cmds->cmd_dep, chip->extra_dma.addr);
	st_le32(&chip->extra_dma.cmds->phy_addr, addr);
	st_le16(&chip->extra_dma.cmds->command, OUTPUT_MORE + BR_ALWAYS);
	out_le32(&chip->awacs->control,
		 (in_le32(&chip->awacs->control) & ~0x1f00)
		 | (speed << 8));
	out_le32(&chip->awacs->byteswap, 0);
	snd_pmac_dma_set_command(rec, &chip->extra_dma);
	snd_pmac_dma_run(rec, RUN);
}

774
void snd_pmac_beep_dma_stop(struct snd_pmac *chip)
L
Linus Torvalds 已提交
775 776 777 778 779 780 781 782 783 784 785
{
	snd_pmac_dma_stop(&chip->playback);
	st_le16(&chip->extra_dma.cmds->command, DBDMA_STOP);
	snd_pmac_pcm_set_format(chip); /* reset format */
}


/*
 * interrupt handlers
 */
static irqreturn_t
786
snd_pmac_tx_intr(int irq, void *devid)
L
Linus Torvalds 已提交
787
{
788
	struct snd_pmac *chip = devid;
L
Linus Torvalds 已提交
789 790 791 792 793 794
	snd_pmac_pcm_update(chip, &chip->playback);
	return IRQ_HANDLED;
}


static irqreturn_t
795
snd_pmac_rx_intr(int irq, void *devid)
L
Linus Torvalds 已提交
796
{
797
	struct snd_pmac *chip = devid;
L
Linus Torvalds 已提交
798 799 800 801 802 803
	snd_pmac_pcm_update(chip, &chip->capture);
	return IRQ_HANDLED;
}


static irqreturn_t
804
snd_pmac_ctrl_intr(int irq, void *devid)
L
Linus Torvalds 已提交
805
{
806
	struct snd_pmac *chip = devid;
L
Linus Torvalds 已提交
807 808 809 810 811 812 813 814 815 816 817 818 819 820 821 822 823 824 825 826 827 828
	int ctrl = in_le32(&chip->awacs->control);

	/*printk("pmac: control interrupt.. 0x%x\n", ctrl);*/
	if (ctrl & MASK_PORTCHG) {
		/* do something when headphone is plugged/unplugged? */
		if (chip->update_automute)
			chip->update_automute(chip, 1);
	}
	if (ctrl & MASK_CNTLERR) {
		int err = (in_le32(&chip->awacs->codec_stat) & MASK_ERRCODE) >> 16;
		if (err && chip->model <= PMAC_SCREAMER)
			snd_printk(KERN_DEBUG "error %x\n", err);
	}
	/* Writing 1s to the CNTLERR and PORTCHG bits clears them... */
	out_le32(&chip->awacs->control, ctrl);
	return IRQ_HANDLED;
}


/*
 * a wrapper to feature call for compatibility
 */
829
static void snd_pmac_sound_feature(struct snd_pmac *chip, int enable)
L
Linus Torvalds 已提交
830
{
831 832
	if (ppc_md.feature_call)
		ppc_md.feature_call(PMAC_FTR_SOUND_CHIP_ENABLE, chip->node, 0, enable);
L
Linus Torvalds 已提交
833 834 835 836 837 838
}

/*
 * release resources
 */

839
static int snd_pmac_free(struct snd_pmac *chip)
L
Linus Torvalds 已提交
840 841 842 843 844 845 846 847
{
	/* stop sounds */
	if (chip->initialized) {
		snd_pmac_dbdma_reset(chip);
		/* disable interrupts from awacs interface */
		out_le32(&chip->awacs->control, in_le32(&chip->awacs->control) & 0xfff);
	}

848 849
	if (chip->node)
		snd_pmac_sound_feature(chip, 0);
L
Linus Torvalds 已提交
850 851 852 853 854 855 856 857 858 859 860 861 862 863

	/* clean up mixer if any */
	if (chip->mixer_free)
		chip->mixer_free(chip);

	snd_pmac_detach_beep(chip);

	/* release resources */
	if (chip->irq >= 0)
		free_irq(chip->irq, (void*)chip);
	if (chip->tx_irq >= 0)
		free_irq(chip->tx_irq, (void*)chip);
	if (chip->rx_irq >= 0)
		free_irq(chip->rx_irq, (void*)chip);
864 865 866
	snd_pmac_dbdma_free(chip, &chip->playback.cmd);
	snd_pmac_dbdma_free(chip, &chip->capture.cmd);
	snd_pmac_dbdma_free(chip, &chip->extra_dma);
867
	snd_pmac_dbdma_free(chip, &emergency_dbdma);
L
Linus Torvalds 已提交
868 869 870 871 872 873 874 875 876 877
	if (chip->macio_base)
		iounmap(chip->macio_base);
	if (chip->latch_base)
		iounmap(chip->latch_base);
	if (chip->awacs)
		iounmap(chip->awacs);
	if (chip->playback.dma)
		iounmap(chip->playback.dma);
	if (chip->capture.dma)
		iounmap(chip->capture.dma);
878

L
Linus Torvalds 已提交
879
	if (chip->node) {
880
		int i;
L
Linus Torvalds 已提交
881
		for (i = 0; i < 3; i++) {
882 883 884 885
			if (chip->requested & (1 << i))
				release_mem_region(chip->rsrc[i].start,
						   chip->rsrc[i].end -
						   chip->rsrc[i].start + 1);
L
Linus Torvalds 已提交
886 887
		}
	}
888

889 890
	if (chip->pdev)
		pci_dev_put(chip->pdev);
891
	of_node_put(chip->node);
L
Linus Torvalds 已提交
892 893 894 895 896 897 898 899
	kfree(chip);
	return 0;
}


/*
 * free the device
 */
900
static int snd_pmac_dev_free(struct snd_device *device)
L
Linus Torvalds 已提交
901
{
902
	struct snd_pmac *chip = device->device_data;
L
Linus Torvalds 已提交
903 904 905 906 907 908 909 910
	return snd_pmac_free(chip);
}


/*
 * check the machine support byteswap (little-endian)
 */

911
static void __init detect_byte_swap(struct snd_pmac *chip)
L
Linus Torvalds 已提交
912 913 914 915 916 917
{
	struct device_node *mio;

	/* if seems that Keylargo can't byte-swap  */
	for (mio = chip->node->parent; mio; mio = mio->parent) {
		if (strcmp(mio->name, "mac-io") == 0) {
918
			if (of_device_is_compatible(mio, "Keylargo"))
L
Linus Torvalds 已提交
919 920 921 922 923 924 925 926 927 928 929 930 931 932 933 934 935 936
				chip->can_byte_swap = 0;
			break;
		}
	}

	/* it seems the Pismo & iBook can't byte-swap in hardware. */
	if (machine_is_compatible("PowerBook3,1") ||
	    machine_is_compatible("PowerBook2,1"))
		chip->can_byte_swap = 0 ;

	if (machine_is_compatible("PowerBook2,1"))
		chip->can_duplex = 0;
}


/*
 * detect a sound chip
 */
937
static int __init snd_pmac_detect(struct snd_pmac *chip)
L
Linus Torvalds 已提交
938
{
939 940
	struct device_node *sound;
	struct device_node *dn;
941 942
	const unsigned int *prop;
	unsigned int l;
943 944
	struct macio_chip* macio;

945
	if (!machine_is(powermac))
L
Linus Torvalds 已提交
946 947 948 949 950 951 952 953 954 955 956
		return -ENODEV;

	chip->subframe = 0;
	chip->revision = 0;
	chip->freqs_ok = 0xff; /* all ok */
	chip->model = PMAC_AWACS;
	chip->can_byte_swap = 1;
	chip->can_duplex = 1;
	chip->can_capture = 1;
	chip->num_freqs = ARRAY_SIZE(awacs_freqs);
	chip->freq_table = awacs_freqs;
957
	chip->pdev = NULL;
L
Linus Torvalds 已提交
958 959 960 961 962 963 964 965 966 967

	chip->control_mask = MASK_IEPC | MASK_IEE | 0x11; /* default */

	/* check machine type */
	if (machine_is_compatible("AAPL,3400/2400")
	    || machine_is_compatible("AAPL,3500"))
		chip->is_pbook_3400 = 1;
	else if (machine_is_compatible("PowerBook1,1")
		 || machine_is_compatible("AAPL,PowerBook1998"))
		chip->is_pbook_G3 = 1;
968 969
	chip->node = of_find_node_by_name(NULL, "awacs");
	sound = of_node_get(chip->node);
L
Linus Torvalds 已提交
970 971 972 973 974

	/*
	 * powermac G3 models have a node called "davbus"
	 * with a child called "sound".
	 */
975
	if (!chip->node)
976
		chip->node = of_find_node_by_name(NULL, "davbus");
L
Linus Torvalds 已提交
977 978 979 980
	/*
	 * if we didn't find a davbus device, try 'i2s-a' since
	 * this seems to be what iBooks have
	 */
981
	if (! chip->node) {
982
		chip->node = of_find_node_by_name(NULL, "i2s-a");
983 984
		if (chip->node && chip->node->parent &&
		    chip->node->parent->parent) {
985
			if (of_device_is_compatible(chip->node->parent->parent,
986 987 988 989
						 "K2-Keylargo"))
				chip->is_k2 = 1;
		}
	}
L
Linus Torvalds 已提交
990 991
	if (! chip->node)
		return -ENODEV;
992

993
	if (!sound) {
994
		sound = of_find_node_by_name(NULL, "sound");
995
		while (sound && sound->parent != chip->node)
996
			sound = of_find_node_by_name(sound, "sound");
997
	}
998 999
	if (! sound) {
		of_node_put(chip->node);
1000
		chip->node = NULL;
L
Linus Torvalds 已提交
1001
		return -ENODEV;
1002
	}
1003
	prop = of_get_property(sound, "sub-frame", NULL);
L
Linus Torvalds 已提交
1004 1005
	if (prop && *prop < 16)
		chip->subframe = *prop;
1006
	prop = of_get_property(sound, "layout-id", NULL);
1007 1008 1009 1010 1011 1012
	if (prop) {
		/* partly deprecate snd-powermac, for those machines
		 * that have a layout-id property for now */
		printk(KERN_INFO "snd-powermac no longer handles any "
				 "machines with a layout-id property "
				 "in the device-tree, use snd-aoa.\n");
1013
		of_node_put(sound);
1014
		of_node_put(chip->node);
1015
		chip->node = NULL;
1016 1017
		return -ENODEV;
	}
L
Linus Torvalds 已提交
1018
	/* This should be verified on older screamers */
1019
	if (of_device_is_compatible(sound, "screamer")) {
L
Linus Torvalds 已提交
1020 1021 1022
		chip->model = PMAC_SCREAMER;
		// chip->can_byte_swap = 0; /* FIXME: check this */
	}
1023
	if (of_device_is_compatible(sound, "burgundy")) {
L
Linus Torvalds 已提交
1024 1025 1026
		chip->model = PMAC_BURGUNDY;
		chip->control_mask = MASK_IEPC | 0x11; /* disable IEE */
	}
1027
	if (of_device_is_compatible(sound, "daca")) {
L
Linus Torvalds 已提交
1028 1029 1030 1031 1032 1033
		chip->model = PMAC_DACA;
		chip->can_capture = 0;  /* no capture */
		chip->can_duplex = 0;
		// chip->can_byte_swap = 0; /* FIXME: check this */
		chip->control_mask = MASK_IEPC | 0x11; /* disable IEE */
	}
1034
	if (of_device_is_compatible(sound, "tumbler")) {
L
Linus Torvalds 已提交
1035 1036 1037 1038 1039 1040 1041 1042
		chip->model = PMAC_TUMBLER;
		chip->can_capture = 0;  /* no capture */
		chip->can_duplex = 0;
		// chip->can_byte_swap = 0; /* FIXME: check this */
		chip->num_freqs = ARRAY_SIZE(tumbler_freqs);
		chip->freq_table = tumbler_freqs;
		chip->control_mask = MASK_IEPC | 0x11; /* disable IEE */
	}
1043
	if (of_device_is_compatible(sound, "snapper")) {
L
Linus Torvalds 已提交
1044 1045 1046 1047 1048 1049
		chip->model = PMAC_SNAPPER;
		// chip->can_byte_swap = 0; /* FIXME: check this */
		chip->num_freqs = ARRAY_SIZE(tumbler_freqs);
		chip->freq_table = tumbler_freqs;
		chip->control_mask = MASK_IEPC | 0x11; /* disable IEE */
	}
1050
	prop = of_get_property(sound, "device-id", NULL);
L
Linus Torvalds 已提交
1051 1052
	if (prop)
		chip->device_id = *prop;
1053 1054 1055
	dn = of_find_node_by_name(NULL, "perch");
	chip->has_iic = (dn != NULL);
	of_node_put(dn);
L
Linus Torvalds 已提交
1056

1057 1058 1059 1060 1061 1062 1063 1064 1065 1066 1067 1068 1069 1070 1071 1072 1073 1074
	/* We need the PCI device for DMA allocations, let's use a crude method
	 * for now ...
	 */
	macio = macio_find(chip->node, macio_unknown);
	if (macio == NULL)
		printk(KERN_WARNING "snd-powermac: can't locate macio !\n");
	else {
		struct pci_dev *pdev = NULL;

		for_each_pci_dev(pdev) {
			struct device_node *np = pci_device_to_OF_node(pdev);
			if (np && np == macio->of_node) {
				chip->pdev = pdev;
				break;
			}
		}
	}
	if (chip->pdev == NULL)
1075 1076
		printk(KERN_WARNING "snd-powermac: can't locate macio PCI"
		       " device !\n");
1077

L
Linus Torvalds 已提交
1078 1079 1080 1081
	detect_byte_swap(chip);

	/* look for a property saying what sample rates
	   are available */
1082
	prop = of_get_property(sound, "sample-rates", &l);
L
Linus Torvalds 已提交
1083
	if (! prop)
1084
		prop = of_get_property(sound, "output-frame-rates", &l);
L
Linus Torvalds 已提交
1085 1086 1087 1088 1089 1090 1091 1092 1093 1094 1095 1096 1097 1098 1099 1100 1101 1102 1103 1104
	if (prop) {
		int i;
		chip->freqs_ok = 0;
		for (l /= sizeof(int); l > 0; --l) {
			unsigned int r = *prop++;
			/* Apple 'Fixed' format */
			if (r >= 0x10000)
				r >>= 16;
			for (i = 0; i < chip->num_freqs; ++i) {
				if (r == chip->freq_table[i]) {
					chip->freqs_ok |= (1 << i);
					break;
				}
			}
		}
	} else {
		/* assume only 44.1khz */
		chip->freqs_ok = 1;
	}

1105
	of_node_put(sound);
L
Linus Torvalds 已提交
1106 1107 1108 1109 1110 1111 1112
	return 0;
}

#ifdef PMAC_SUPPORT_AUTOMUTE
/*
 * auto-mute
 */
1113 1114
static int pmac_auto_mute_get(struct snd_kcontrol *kcontrol,
			      struct snd_ctl_elem_value *ucontrol)
L
Linus Torvalds 已提交
1115
{
1116
	struct snd_pmac *chip = snd_kcontrol_chip(kcontrol);
L
Linus Torvalds 已提交
1117 1118 1119 1120
	ucontrol->value.integer.value[0] = chip->auto_mute;
	return 0;
}

1121 1122
static int pmac_auto_mute_put(struct snd_kcontrol *kcontrol,
			      struct snd_ctl_elem_value *ucontrol)
L
Linus Torvalds 已提交
1123
{
1124
	struct snd_pmac *chip = snd_kcontrol_chip(kcontrol);
L
Linus Torvalds 已提交
1125
	if (ucontrol->value.integer.value[0] != chip->auto_mute) {
1126
		chip->auto_mute = !!ucontrol->value.integer.value[0];
L
Linus Torvalds 已提交
1127 1128 1129 1130 1131 1132 1133
		if (chip->update_automute)
			chip->update_automute(chip, 1);
		return 1;
	}
	return 0;
}

1134 1135
static int pmac_hp_detect_get(struct snd_kcontrol *kcontrol,
			      struct snd_ctl_elem_value *ucontrol)
L
Linus Torvalds 已提交
1136
{
1137
	struct snd_pmac *chip = snd_kcontrol_chip(kcontrol);
L
Linus Torvalds 已提交
1138 1139 1140 1141 1142 1143 1144
	if (chip->detect_headphone)
		ucontrol->value.integer.value[0] = chip->detect_headphone(chip);
	else
		ucontrol->value.integer.value[0] = 0;
	return 0;
}

1145
static struct snd_kcontrol_new auto_mute_controls[] __initdata = {
L
Linus Torvalds 已提交
1146 1147 1148 1149 1150 1151 1152 1153 1154 1155 1156 1157 1158 1159
	{ .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
	  .name = "Auto Mute Switch",
	  .info = snd_pmac_boolean_mono_info,
	  .get = pmac_auto_mute_get,
	  .put = pmac_auto_mute_put,
	},
	{ .iface = SNDRV_CTL_ELEM_IFACE_MIXER,
	  .name = "Headphone Detection",
	  .access = SNDRV_CTL_ELEM_ACCESS_READ,
	  .info = snd_pmac_boolean_mono_info,
	  .get = pmac_hp_detect_get,
	},
};

1160
int __init snd_pmac_add_automute(struct snd_pmac *chip)
L
Linus Torvalds 已提交
1161 1162 1163 1164
{
	int err;
	chip->auto_mute = 1;
	err = snd_ctl_add(chip->card, snd_ctl_new1(&auto_mute_controls[0], chip));
1165 1166
	if (err < 0) {
		printk(KERN_ERR "snd-powermac: Failed to add automute control\n");
L
Linus Torvalds 已提交
1167
		return err;
1168
	}
L
Linus Torvalds 已提交
1169 1170 1171 1172 1173 1174 1175 1176
	chip->hp_detect_ctl = snd_ctl_new1(&auto_mute_controls[1], chip);
	return snd_ctl_add(chip->card, chip->hp_detect_ctl);
}
#endif /* PMAC_SUPPORT_AUTOMUTE */

/*
 * create and detect a pmac chip record
 */
1177
int __init snd_pmac_new(struct snd_card *card, struct snd_pmac **chip_return)
L
Linus Torvalds 已提交
1178
{
1179
	struct snd_pmac *chip;
L
Linus Torvalds 已提交
1180 1181
	struct device_node *np;
	int i, err;
1182
	unsigned int irq;
1183
	unsigned long ctrl_addr, txdma_addr, rxdma_addr;
1184
	static struct snd_device_ops ops = {
L
Linus Torvalds 已提交
1185 1186 1187 1188 1189
		.dev_free =	snd_pmac_dev_free,
	};

	*chip_return = NULL;

1190
	chip = kzalloc(sizeof(*chip), GFP_KERNEL);
L
Linus Torvalds 已提交
1191 1192 1193 1194 1195 1196 1197 1198 1199 1200 1201 1202 1203
	if (chip == NULL)
		return -ENOMEM;
	chip->card = card;

	spin_lock_init(&chip->reg_lock);
	chip->irq = chip->tx_irq = chip->rx_irq = -1;

	chip->playback.stream = SNDRV_PCM_STREAM_PLAYBACK;
	chip->capture.stream = SNDRV_PCM_STREAM_CAPTURE;

	if ((err = snd_pmac_detect(chip)) < 0)
		goto __error;

1204 1205
	if (snd_pmac_dbdma_alloc(chip, &chip->playback.cmd, PMAC_MAX_FRAGS + 1) < 0 ||
	    snd_pmac_dbdma_alloc(chip, &chip->capture.cmd, PMAC_MAX_FRAGS + 1) < 0 ||
1206 1207
	    snd_pmac_dbdma_alloc(chip, &chip->extra_dma, 2) < 0 ||
	    snd_pmac_dbdma_alloc(chip, &emergency_dbdma, 2) < 0) {
L
Linus Torvalds 已提交
1208 1209 1210 1211 1212
		err = -ENOMEM;
		goto __error;
	}

	np = chip->node;
1213
	chip->requested = 0;
1214
	if (chip->is_k2) {
1215 1216 1217 1218 1219 1220 1221 1222 1223 1224 1225 1226 1227 1228 1229
		static char *rnames[] = {
			"Sound Control", "Sound DMA" };
		for (i = 0; i < 2; i ++) {
			if (of_address_to_resource(np->parent, i,
						   &chip->rsrc[i])) {
				printk(KERN_ERR "snd: can't translate rsrc "
				       " %d (%s)\n", i, rnames[i]);
				err = -ENODEV;
				goto __error;
			}
			if (request_mem_region(chip->rsrc[i].start,
					       chip->rsrc[i].end -
					       chip->rsrc[i].start + 1,
					       rnames[i]) == NULL) {
				printk(KERN_ERR "snd: can't request rsrc "
1230
				       " %d (%s: 0x%016llx:%016llx)\n",
1231 1232 1233
				       i, rnames[i],
				       (unsigned long long)chip->rsrc[i].start,
				       (unsigned long long)chip->rsrc[i].end);
1234 1235 1236
				err = -ENODEV;
				goto __error;
			}
1237
			chip->requested |= (1 << i);
1238
		}
1239 1240 1241
		ctrl_addr = chip->rsrc[0].start;
		txdma_addr = chip->rsrc[1].start;
		rxdma_addr = txdma_addr + 0x100;
1242
	} else {
1243 1244 1245
		static char *rnames[] = {
			"Sound Control", "Sound Tx DMA", "Sound Rx DMA" };
		for (i = 0; i < 3; i ++) {
1246
			if (of_address_to_resource(np, i,
1247 1248 1249 1250 1251 1252 1253 1254 1255 1256 1257
						   &chip->rsrc[i])) {
				printk(KERN_ERR "snd: can't translate rsrc "
				       " %d (%s)\n", i, rnames[i]);
				err = -ENODEV;
				goto __error;
			}
			if (request_mem_region(chip->rsrc[i].start,
					       chip->rsrc[i].end -
					       chip->rsrc[i].start + 1,
					       rnames[i]) == NULL) {
				printk(KERN_ERR "snd: can't request rsrc "
1258 1259 1260 1261
				       " %d (%s: 0x%016llx:%016llx)\n",
				       i, rnames[i],
				       (unsigned long long)chip->rsrc[i].start,
				       (unsigned long long)chip->rsrc[i].end);
1262 1263 1264
				err = -ENODEV;
				goto __error;
			}
1265
			chip->requested |= (1 << i);
1266
		}
1267 1268 1269
		ctrl_addr = chip->rsrc[0].start;
		txdma_addr = chip->rsrc[1].start;
		rxdma_addr = chip->rsrc[2].start;
L
Linus Torvalds 已提交
1270 1271
	}

1272 1273 1274
	chip->awacs = ioremap(ctrl_addr, 0x1000);
	chip->playback.dma = ioremap(txdma_addr, 0x100);
	chip->capture.dma = ioremap(rxdma_addr, 0x100);
L
Linus Torvalds 已提交
1275
	if (chip->model <= PMAC_BURGUNDY) {
1276 1277
		irq = irq_of_parse_and_map(np, 0);
		if (request_irq(irq, snd_pmac_ctrl_intr, 0,
L
Linus Torvalds 已提交
1278
				"PMac", (void*)chip)) {
1279 1280
			snd_printk(KERN_ERR "pmac: unable to grab IRQ %d\n",
				   irq);
L
Linus Torvalds 已提交
1281 1282 1283
			err = -EBUSY;
			goto __error;
		}
1284
		chip->irq = irq;
L
Linus Torvalds 已提交
1285
	}
1286 1287 1288
	irq = irq_of_parse_and_map(np, 1);
	if (request_irq(irq, snd_pmac_tx_intr, 0, "PMac Output", (void*)chip)){
		snd_printk(KERN_ERR "pmac: unable to grab IRQ %d\n", irq);
L
Linus Torvalds 已提交
1289 1290 1291
		err = -EBUSY;
		goto __error;
	}
1292 1293 1294 1295
	chip->tx_irq = irq;
	irq = irq_of_parse_and_map(np, 2);
	if (request_irq(irq, snd_pmac_rx_intr, 0, "PMac Input", (void*)chip)) {
		snd_printk(KERN_ERR "pmac: unable to grab IRQ %d\n", irq);
L
Linus Torvalds 已提交
1296 1297 1298
		err = -EBUSY;
		goto __error;
	}
1299
	chip->rx_irq = irq;
L
Linus Torvalds 已提交
1300 1301 1302

	snd_pmac_sound_feature(chip, 1);

1303 1304 1305
	/* reset & enable interrupts */
	if (chip->model <= PMAC_BURGUNDY)
		out_le32(&chip->awacs->control, chip->control_mask);
L
Linus Torvalds 已提交
1306 1307 1308 1309 1310 1311 1312 1313 1314 1315 1316 1317 1318 1319 1320 1321 1322

	/* Powerbooks have odd ways of enabling inputs such as
	   an expansion-bay CD or sound from an internal modem
	   or a PC-card modem. */
	if (chip->is_pbook_3400) {
		/* Enable CD and PC-card sound inputs. */
		/* This is done by reading from address
		 * f301a000, + 0x10 to enable the expansion-bay
		 * CD sound input, + 0x80 to enable the PC-card
		 * sound input.  The 0x100 enables the SCSI bus
		 * terminator power.
		 */
		chip->latch_base = ioremap (0xf301a000, 0x1000);
		in_8(chip->latch_base + 0x190);
	} else if (chip->is_pbook_G3) {
		struct device_node* mio;
		for (mio = chip->node->parent; mio; mio = mio->parent) {
1323 1324 1325 1326 1327
			if (strcmp(mio->name, "mac-io") == 0) {
				struct resource r;
				if (of_address_to_resource(mio, 0, &r) == 0)
					chip->macio_base =
						ioremap(r.start, 0x40);
L
Linus Torvalds 已提交
1328 1329 1330 1331 1332 1333 1334 1335 1336 1337 1338 1339 1340 1341 1342 1343 1344 1345 1346 1347 1348 1349 1350 1351 1352 1353 1354 1355 1356 1357 1358 1359 1360 1361
				break;
			}
		}
		/* Enable CD sound input. */
		/* The relevant bits for writing to this byte are 0x8f.
		 * I haven't found out what the 0x80 bit does.
		 * For the 0xf bits, writing 3 or 7 enables the CD
		 * input, any other value disables it.  Values
		 * 1, 3, 5, 7 enable the microphone.  Values 0, 2,
		 * 4, 6, 8 - f enable the input from the modem.
		 */
		if (chip->macio_base)
			out_8(chip->macio_base + 0x37, 3);
	}

	/* Reset dbdma channels */
	snd_pmac_dbdma_reset(chip);

	if ((err = snd_device_new(card, SNDRV_DEV_LOWLEVEL, chip, &ops)) < 0)
		goto __error;

	*chip_return = chip;
	return 0;

 __error:
	snd_pmac_free(chip);
	return err;
}


/*
 * sleep notify for powerbook
 */

1362
#ifdef CONFIG_PM
L
Linus Torvalds 已提交
1363 1364 1365 1366 1367

/*
 * Save state when going to sleep, restore it afterwards.
 */

1368
void snd_pmac_suspend(struct snd_pmac *chip)
L
Linus Torvalds 已提交
1369 1370 1371
{
	unsigned long flags;

1372
	snd_power_change_state(chip->card, SNDRV_CTL_POWER_D3hot);
L
Linus Torvalds 已提交
1373 1374 1375 1376 1377 1378 1379 1380 1381 1382 1383 1384 1385 1386 1387
	if (chip->suspend)
		chip->suspend(chip);
	snd_pcm_suspend_all(chip->pcm);
	spin_lock_irqsave(&chip->reg_lock, flags);
	snd_pmac_beep_stop(chip);
	spin_unlock_irqrestore(&chip->reg_lock, flags);
	if (chip->irq >= 0)
		disable_irq(chip->irq);
	if (chip->tx_irq >= 0)
		disable_irq(chip->tx_irq);
	if (chip->rx_irq >= 0)
		disable_irq(chip->rx_irq);
	snd_pmac_sound_feature(chip, 0);
}

1388
void snd_pmac_resume(struct snd_pmac *chip)
L
Linus Torvalds 已提交
1389 1390 1391 1392 1393
{
	snd_pmac_sound_feature(chip, 1);
	if (chip->resume)
		chip->resume(chip);
	/* enable CD sound input */
1394
	if (chip->macio_base && chip->is_pbook_G3)
L
Linus Torvalds 已提交
1395
		out_8(chip->macio_base + 0x37, 3);
1396
	else if (chip->is_pbook_3400)
L
Linus Torvalds 已提交
1397 1398 1399 1400 1401 1402 1403 1404 1405 1406 1407
		in_8(chip->latch_base + 0x190);

	snd_pmac_pcm_set_format(chip);

	if (chip->irq >= 0)
		enable_irq(chip->irq);
	if (chip->tx_irq >= 0)
		enable_irq(chip->tx_irq);
	if (chip->rx_irq >= 0)
		enable_irq(chip->rx_irq);

1408
	snd_power_change_state(chip->card, SNDRV_CTL_POWER_D0);
L
Linus Torvalds 已提交
1409 1410
}

1411 1412
#endif /* CONFIG_PM */