xref: /freebsd/sys/dev/sound/pci/envy24ht.c (revision 45dd2eaac379e5576f745380260470204c49beac)
1 /*-
2  * SPDX-License-Identifier: BSD-2-Clause-FreeBSD
3  *
4  * Copyright (c) 2006 Konstantin Dimitrov <kosio.dimitrov@gmail.com>
5  * Copyright (c) 2001 Katsurajima Naoto <raven@katsurajima.seya.yokohama.jp>
6  * All rights reserved.
7  *
8  * Redistribution and use in source and binary forms, with or without
9  * modification, are permitted provided that the following conditions
10  * are met:
11  * 1. Redistributions of source code must retain the above copyright
12  *    notice, this list of conditions and the following disclaimer.
13  * 2. Redistributions in binary form must reproduce the above copyright
14  *    notice, this list of conditions and the following disclaimer in the
15  *    documentation and/or other materials provided with the distribution.
16  *
17  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
18  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
21  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
23  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
24  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHERIN CONTRACT, STRICT
25  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
26  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
27  * SUCH DAMAGE.
28  *
29  */
30 
31 /*
32  * Konstantin Dimitrov's thanks list:
33  *
34  * A huge thanks goes to Spas Filipov for his friendship, support and his
35  * generous gift - an 'Audiotrak Prodigy HD2' audio card! I also want to
36  * thank Keiichi Iwasaki and his parents, because they helped Spas to get
37  * the card from Japan! Having hardware sample of Prodigy HD2 made adding
38  * support for that great card very easy and real fun and pleasure.
39  *
40  */
41 
42 #ifdef HAVE_KERNEL_OPTION_HEADERS
43 #include "opt_snd.h"
44 #endif
45 
46 #include <dev/sound/pcm/sound.h>
47 #include <dev/sound/pcm/ac97.h>
48 #include <dev/sound/pci/spicds.h>
49 #include <dev/sound/pci/envy24ht.h>
50 
51 #include <dev/pci/pcireg.h>
52 #include <dev/pci/pcivar.h>
53 
54 #include "mixer_if.h"
55 
56 SND_DECLARE_FILE("$FreeBSD$");
57 
58 static MALLOC_DEFINE(M_ENVY24HT, "envy24ht", "envy24ht audio");
59 
60 /* -------------------------------------------------------------------- */
61 
62 struct sc_info;
63 
64 #define ENVY24HT_PLAY_CHNUM 8
65 #define ENVY24HT_REC_CHNUM 2
66 #define ENVY24HT_PLAY_BUFUNIT (4 /* byte/sample */ * 8 /* channel */)
67 #define ENVY24HT_REC_BUFUNIT  (4 /* byte/sample */ * 2 /* channel */)
68 #define ENVY24HT_SAMPLE_NUM   4096
69 
70 #define ENVY24HT_TIMEOUT 1000
71 
72 #define ENVY24HT_DEFAULT_FORMAT	SND_FORMAT(AFMT_S16_LE, 2, 0)
73 
74 #define ENVY24HT_NAMELEN 32
75 
76 struct envy24ht_sample {
77         volatile u_int32_t buffer;
78 };
79 
80 typedef struct envy24ht_sample sample32_t;
81 
82 /* channel registers */
83 struct sc_chinfo {
84 	struct snd_dbuf		*buffer;
85 	struct pcm_channel	*channel;
86 	struct sc_info		*parent;
87 	int			dir;
88 	unsigned		num; /* hw channel number */
89 
90 	/* channel information */
91 	u_int32_t		format;
92 	u_int32_t		speed;
93 	u_int32_t		blk; /* hw block size(dword) */
94 
95 	/* format conversion structure */
96 	u_int8_t		*data;
97 	unsigned int		size; /* data buffer size(byte) */
98 	int			unit; /* sample size(byte) */
99 	unsigned int		offset; /* samples number offset */
100 	void			(*emldma)(struct sc_chinfo *);
101 
102 	/* flags */
103 	int			run;
104 };
105 
106 /* codec interface entrys */
107 struct codec_entry {
108 	void *(*create)(device_t dev, void *devinfo, int dir, int num);
109 	void (*destroy)(void *codec);
110 	void (*init)(void *codec);
111 	void (*reinit)(void *codec);
112 	void (*setvolume)(void *codec, int dir, unsigned int left, unsigned int right);
113 	void (*setrate)(void *codec, int which, int rate);
114 };
115 
116 /* system configuration information */
117 struct cfg_info {
118 	char *name;
119 	u_int16_t subvendor, subdevice;
120 	u_int8_t scfg, acl, i2s, spdif;
121 	u_int32_t gpiomask, gpiostate, gpiodir;
122 	u_int32_t cdti, cclk, cs;
123 	u_int8_t cif, type, free;
124 	struct codec_entry *codec;
125 };
126 
127 /* device private data */
128 struct sc_info {
129 	device_t	dev;
130 	struct mtx	*lock;
131 
132 	/* Control/Status registor */
133 	struct resource *cs;
134 	int		csid;
135 	bus_space_tag_t cst;
136 	bus_space_handle_t csh;
137 	/* MultiTrack registor */
138 	struct resource *mt;
139 	int		mtid;
140 	bus_space_tag_t mtt;
141 	bus_space_handle_t mth;
142 	/* DMA tag */
143 	bus_dma_tag_t dmat;
144 	/* IRQ resource */
145 	struct resource *irq;
146 	int		irqid;
147 	void		*ih;
148 
149 	/* system configuration data */
150 	struct cfg_info *cfg;
151 
152 	/* ADC/DAC number and info */
153 	int		adcn, dacn;
154 	void		*adc[4], *dac[4];
155 
156 	/* mixer control data */
157 	u_int32_t	src;
158 	u_int8_t	left[ENVY24HT_CHAN_NUM];
159 	u_int8_t	right[ENVY24HT_CHAN_NUM];
160 
161 	/* Play/Record DMA fifo */
162 	sample32_t	*pbuf;
163 	sample32_t	*rbuf;
164 	u_int32_t	psize, rsize; /* DMA buffer size(byte) */
165 	u_int16_t	blk[2]; /* transfer check blocksize(dword) */
166 	bus_dmamap_t	pmap, rmap;
167 	bus_addr_t	paddr, raddr;
168 
169 	/* current status */
170 	u_int32_t	speed;
171 	int		run[2];
172 	u_int16_t	intr[2];
173 	struct pcmchan_caps	caps[2];
174 
175 	/* channel info table */
176 	unsigned	chnum;
177 	struct sc_chinfo chan[11];
178 };
179 
180 /* -------------------------------------------------------------------- */
181 
182 /*
183  * prototypes
184  */
185 
186 /* DMA emulator */
187 static void envy24ht_p8u(struct sc_chinfo *);
188 static void envy24ht_p16sl(struct sc_chinfo *);
189 static void envy24ht_p32sl(struct sc_chinfo *);
190 static void envy24ht_r16sl(struct sc_chinfo *);
191 static void envy24ht_r32sl(struct sc_chinfo *);
192 
193 /* channel interface */
194 static void *envy24htchan_init(kobj_t, void *, struct snd_dbuf *, struct pcm_channel *, int);
195 static int envy24htchan_setformat(kobj_t, void *, u_int32_t);
196 static u_int32_t envy24htchan_setspeed(kobj_t, void *, u_int32_t);
197 static u_int32_t envy24htchan_setblocksize(kobj_t, void *, u_int32_t);
198 static int envy24htchan_trigger(kobj_t, void *, int);
199 static u_int32_t envy24htchan_getptr(kobj_t, void *);
200 static struct pcmchan_caps *envy24htchan_getcaps(kobj_t, void *);
201 
202 /* mixer interface */
203 static int envy24htmixer_init(struct snd_mixer *);
204 static int envy24htmixer_reinit(struct snd_mixer *);
205 static int envy24htmixer_uninit(struct snd_mixer *);
206 static int envy24htmixer_set(struct snd_mixer *, unsigned, unsigned, unsigned);
207 static u_int32_t envy24htmixer_setrecsrc(struct snd_mixer *, u_int32_t);
208 
209 /* SPI codec access interface */
210 static void *envy24ht_spi_create(device_t, void *, int, int);
211 static void envy24ht_spi_destroy(void *);
212 static void envy24ht_spi_init(void *);
213 static void envy24ht_spi_reinit(void *);
214 static void envy24ht_spi_setvolume(void *, int, unsigned int, unsigned int);
215 
216 /* -------------------------------------------------------------------- */
217 
218 /*
219   system constant tables
220 */
221 
222 /* API -> hardware channel map */
223 static unsigned envy24ht_chanmap[ENVY24HT_CHAN_NUM] = {
224 	ENVY24HT_CHAN_PLAY_DAC1,  /* 1 */
225 	ENVY24HT_CHAN_PLAY_DAC2,  /* 2 */
226 	ENVY24HT_CHAN_PLAY_DAC3,  /* 3 */
227 	ENVY24HT_CHAN_PLAY_DAC4,  /* 4 */
228 	ENVY24HT_CHAN_PLAY_SPDIF, /* 0 */
229 	ENVY24HT_CHAN_REC_MIX,    /* 5 */
230 	ENVY24HT_CHAN_REC_SPDIF,  /* 6 */
231 	ENVY24HT_CHAN_REC_ADC1,   /* 7 */
232 	ENVY24HT_CHAN_REC_ADC2,   /* 8 */
233 	ENVY24HT_CHAN_REC_ADC3,   /* 9 */
234 	ENVY24HT_CHAN_REC_ADC4,   /* 10 */
235 };
236 
237 /* mixer -> API channel map. see above */
238 static int envy24ht_mixmap[] = {
239 	-1, /* Master output level. It is depend on codec support */
240 	-1, /* Treble level of all output channels */
241 	-1, /* Bass level of all output channels */
242 	-1, /* Volume of synthesier input */
243 	0,  /* Output level for the audio device */
244 	-1, /* Output level for the PC speaker */
245 	7,  /* line in jack */
246 	-1, /* microphone jack */
247 	-1, /* CD audio input */
248 	-1, /* Recording monitor */
249 	1,  /* alternative codec */
250 	-1, /* global recording level */
251 	-1, /* Input gain */
252 	-1, /* Output gain */
253 	8,  /* Input source 1 */
254 	9,  /* Input source 2 */
255 	10, /* Input source 3 */
256 	6,  /* Digital (input) 1 */
257 	-1, /* Digital (input) 2 */
258 	-1, /* Digital (input) 3 */
259 	-1, /* Phone input */
260 	-1, /* Phone output */
261 	-1, /* Video/TV (audio) in */
262 	-1, /* Radio in */
263 	-1, /* Monitor volume */
264 };
265 
266 /* variable rate audio */
267 static u_int32_t envy24ht_speed[] = {
268     192000, 176400, 96000, 88200, 64000, 48000, 44100, 32000, 24000, 22050, 16000,
269     12000, 11025, 9600, 8000, 0
270 };
271 
272 /* known boards configuration */
273 static struct codec_entry spi_codec = {
274 	envy24ht_spi_create,
275 	envy24ht_spi_destroy,
276 	envy24ht_spi_init,
277 	envy24ht_spi_reinit,
278 	envy24ht_spi_setvolume,
279 	NULL, /* setrate */
280 };
281 
282 static struct cfg_info cfg_table[] = {
283 	{
284 		"Envy24HT audio (Terratec Aureon 7.1 Space)",
285 		0x153b, 0x1145,
286 		0x0b, 0x80, 0xfc, 0xc3,
287 		0x21efff, 0x7fffff, 0x5e1000,
288 		0x40000, 0x80000, 0x1000, 0x00, 0x02,
289 		0,
290 		&spi_codec,
291 	},
292         {
293                 "Envy24HT audio (Terratec Aureon 5.1 Sky)",
294                 0x153b, 0x1147,
295                 0x0a, 0x80, 0xfc, 0xc3,
296                 0x21efff, 0x7fffff, 0x5e1000,
297                 0x40000, 0x80000, 0x1000, 0x00, 0x02,
298                 0,
299                 &spi_codec,
300         },
301 	        {
302                 "Envy24HT audio (Terratec Aureon 7.1 Universe)",
303                 0x153b, 0x1153,
304                 0x0b, 0x80, 0xfc, 0xc3,
305                 0x21efff, 0x7fffff, 0x5e1000,
306                 0x40000, 0x80000, 0x1000, 0x00, 0x02,
307                 0,
308                 &spi_codec,
309         },
310         {
311                 "Envy24HT audio (AudioTrak Prodigy 7.1)",
312                 0x4933, 0x4553,
313                 0x0b, 0x80, 0xfc, 0xc3,
314                 0x21efff, 0x7fffff, 0x5e1000,
315                 0x40000, 0x80000, 0x1000, 0x00, 0x02,
316                 0,
317                 &spi_codec,
318         },
319         {
320                 "Envy24HT audio (Terratec PHASE 28)",
321                 0x153b, 0x1149,
322                 0x0b, 0x80, 0xfc, 0xc3,
323                 0x21efff, 0x7fffff, 0x5e1000,
324                 0x40000, 0x80000, 0x1000, 0x00, 0x02,
325                 0,
326                 &spi_codec,
327         },
328         {
329                 "Envy24HT-S audio (Terratec PHASE 22)",
330                 0x153b, 0x1150,
331                 0x10, 0x80, 0xf0, 0xc3,
332                 0x7ffbc7, 0x7fffff, 0x438,
333                 0x10, 0x20, 0x400, 0x01, 0x00,
334                 0,
335                 &spi_codec,
336         },
337         {
338                 "Envy24HT audio (AudioTrak Prodigy 7.1 LT)",
339                 0x3132, 0x4154,
340                 0x4b, 0x80, 0xfc, 0xc3,
341                 0x7ff8ff, 0x7fffff, 0x700,
342                 0x400, 0x200, 0x100, 0x00, 0x02,
343                 0,
344                 &spi_codec,
345         },
346         {
347                 "Envy24HT audio (AudioTrak Prodigy 7.1 XT)",
348                 0x3136, 0x4154,
349                 0x4b, 0x80, 0xfc, 0xc3,
350                 0x7ff8ff, 0x7fffff, 0x700,
351                 0x400, 0x200, 0x100, 0x00, 0x02,
352                 0,
353                 &spi_codec,
354         },
355         {
356                 "Envy24HT audio (M-Audio Revolution 7.1)",
357                 0x1412, 0x3630,
358                 0x43, 0x80, 0xf8, 0xc1,
359                 0x3fff85, 0x400072, 0x4000fa,
360                 0x08, 0x02, 0x20, 0x00, 0x04,
361                 0,
362                 &spi_codec,
363         },
364         {
365                 "Envy24GT audio (M-Audio Revolution 5.1)",
366                 0x1412, 0x3631,
367                 0x42, 0x80, 0xf8, 0xc1,
368                 0x3fff05, 0x4000f0, 0x4000fa,
369                 0x08, 0x02, 0x10, 0x00, 0x03,
370                 0,
371                 &spi_codec,
372         },
373         {
374                 "Envy24HT audio (M-Audio Audiophile 192)",
375                 0x1412, 0x3632,
376                 0x68, 0x80, 0xf8, 0xc3,
377                 0x45, 0x4000b5, 0x7fffba,
378                 0x08, 0x02, 0x10, 0x00, 0x03,
379                 0,
380                 &spi_codec,
381         },
382         {
383                 "Envy24HT audio (AudioTrak Prodigy HD2)",
384                 0x3137, 0x4154,
385                 0x68, 0x80, 0x78, 0xc3,
386                 0xfff8ff, 0x200700, 0xdfffff,
387                 0x400, 0x200, 0x100, 0x00, 0x05,
388                 0,
389                 &spi_codec,
390         },
391         {
392                 "Envy24HT audio (ESI Juli@)",
393                 0x3031, 0x4553,
394                 0x20, 0x80, 0xf8, 0xc3,
395                 0x7fff9f, 0x8016, 0x7fff9f,
396                 0x08, 0x02, 0x10, 0x00, 0x03,
397                 0,
398                 &spi_codec,
399         },
400 	{
401                 "Envy24HT-S audio (Terrasoniq TS22PCI)",
402                 0x153b, 0x117b,
403                 0x10, 0x80, 0xf0, 0xc3,
404                 0x7ffbc7, 0x7fffff, 0x438,
405                 0x10, 0x20, 0x400, 0x01, 0x00,
406                 0,
407                 &spi_codec,
408 	},
409 	{
410 		"Envy24HT audio (Generic)",
411 		0, 0,
412 		0x0b, 0x80, 0xfc, 0xc3,
413 		0x21efff, 0x7fffff, 0x5e1000,
414                 0x40000, 0x80000, 0x1000, 0x00, 0x02,
415 		0,
416 		&spi_codec, /* default codec routines */
417 	}
418 };
419 
420 static u_int32_t envy24ht_recfmt[] = {
421 	SND_FORMAT(AFMT_S16_LE, 2, 0),
422 	SND_FORMAT(AFMT_S32_LE, 2, 0),
423 	0
424 };
425 static struct pcmchan_caps envy24ht_reccaps = {8000, 96000, envy24ht_recfmt, 0};
426 
427 static u_int32_t envy24ht_playfmt[] = {
428 	SND_FORMAT(AFMT_U8, 2, 0),
429 	SND_FORMAT(AFMT_S16_LE, 2, 0),
430 	SND_FORMAT(AFMT_S32_LE, 2, 0),
431 	0
432 };
433 
434 static struct pcmchan_caps envy24ht_playcaps = {8000, 192000, envy24ht_playfmt, 0};
435 
436 struct envy24ht_emldma {
437 	u_int32_t	format;
438 	void		(*emldma)(struct sc_chinfo *);
439 	int		unit;
440 };
441 
442 static struct envy24ht_emldma envy24ht_pemltab[] = {
443 	{SND_FORMAT(AFMT_U8, 2, 0), envy24ht_p8u, 2},
444 	{SND_FORMAT(AFMT_S16_LE, 2, 0), envy24ht_p16sl, 4},
445 	{SND_FORMAT(AFMT_S32_LE, 2, 0), envy24ht_p32sl, 8},
446 	{0, NULL, 0}
447 };
448 
449 static struct envy24ht_emldma envy24ht_remltab[] = {
450 	{SND_FORMAT(AFMT_S16_LE, 2, 0), envy24ht_r16sl, 4},
451 	{SND_FORMAT(AFMT_S32_LE, 2, 0), envy24ht_r32sl, 8},
452 	{0, NULL, 0}
453 };
454 
455 /* -------------------------------------------------------------------- */
456 
457 /* common routines */
458 static u_int32_t
459 envy24ht_rdcs(struct sc_info *sc, int regno, int size)
460 {
461 	switch (size) {
462 	case 1:
463 		return bus_space_read_1(sc->cst, sc->csh, regno);
464 	case 2:
465 		return bus_space_read_2(sc->cst, sc->csh, regno);
466 	case 4:
467 		return bus_space_read_4(sc->cst, sc->csh, regno);
468 	default:
469 		return 0xffffffff;
470 	}
471 }
472 
473 static void
474 envy24ht_wrcs(struct sc_info *sc, int regno, u_int32_t data, int size)
475 {
476 	switch (size) {
477 	case 1:
478 		bus_space_write_1(sc->cst, sc->csh, regno, data);
479 		break;
480 	case 2:
481 		bus_space_write_2(sc->cst, sc->csh, regno, data);
482 		break;
483 	case 4:
484 		bus_space_write_4(sc->cst, sc->csh, regno, data);
485 		break;
486 	}
487 }
488 
489 static u_int32_t
490 envy24ht_rdmt(struct sc_info *sc, int regno, int size)
491 {
492 	switch (size) {
493 	case 1:
494 		return bus_space_read_1(sc->mtt, sc->mth, regno);
495 	case 2:
496 		return bus_space_read_2(sc->mtt, sc->mth, regno);
497 	case 4:
498 		return bus_space_read_4(sc->mtt, sc->mth, regno);
499 	default:
500 		return 0xffffffff;
501 	}
502 }
503 
504 static void
505 envy24ht_wrmt(struct sc_info *sc, int regno, u_int32_t data, int size)
506 {
507 	switch (size) {
508 	case 1:
509 		bus_space_write_1(sc->mtt, sc->mth, regno, data);
510 		break;
511 	case 2:
512 		bus_space_write_2(sc->mtt, sc->mth, regno, data);
513 		break;
514 	case 4:
515 		bus_space_write_4(sc->mtt, sc->mth, regno, data);
516 		break;
517 	}
518 }
519 
520 /* -------------------------------------------------------------------- */
521 
522 /* I2C port/E2PROM access routines */
523 
524 static int
525 envy24ht_rdi2c(struct sc_info *sc, u_int32_t dev, u_int32_t addr)
526 {
527 	u_int32_t data;
528 	int i;
529 
530 #if(0)
531 	device_printf(sc->dev, "envy24ht_rdi2c(sc, 0x%02x, 0x%02x)\n", dev, addr);
532 #endif
533 	for (i = 0; i < ENVY24HT_TIMEOUT; i++) {
534 		data = envy24ht_rdcs(sc, ENVY24HT_CCS_I2CSTAT, 1);
535 		if ((data & ENVY24HT_CCS_I2CSTAT_BSY) == 0)
536 			break;
537 		DELAY(32); /* 31.25kHz */
538 	}
539 	if (i == ENVY24HT_TIMEOUT) {
540 		return -1;
541 	}
542 	envy24ht_wrcs(sc, ENVY24HT_CCS_I2CADDR, addr, 1);
543 	envy24ht_wrcs(sc, ENVY24HT_CCS_I2CDEV,
544 	    (dev & ENVY24HT_CCS_I2CDEV_ADDR) | ENVY24HT_CCS_I2CDEV_RD, 1);
545 	for (i = 0; i < ENVY24HT_TIMEOUT; i++) {
546 		data = envy24ht_rdcs(sc, ENVY24HT_CCS_I2CSTAT, 1);
547 		if ((data & ENVY24HT_CCS_I2CSTAT_BSY) == 0)
548 			break;
549 		DELAY(32); /* 31.25kHz */
550 	}
551 	if (i == ENVY24HT_TIMEOUT) {
552 		return -1;
553 	}
554 	data = envy24ht_rdcs(sc, ENVY24HT_CCS_I2CDATA, 1);
555 
556 #if(0)
557 	device_printf(sc->dev, "envy24ht_rdi2c(): return 0x%x\n", data);
558 #endif
559 	return (int)data;
560 }
561 
562 static int
563 envy24ht_wri2c(struct sc_info *sc, u_int32_t dev, u_int32_t addr, u_int32_t data)
564 {
565 	u_int32_t tmp;
566 	int i;
567 
568 #if(0)
569 	device_printf(sc->dev, "envy24ht_rdi2c(sc, 0x%02x, 0x%02x)\n", dev, addr);
570 #endif
571 	for (i = 0; i < ENVY24HT_TIMEOUT; i++) {
572 		tmp = envy24ht_rdcs(sc, ENVY24HT_CCS_I2CSTAT, 1);
573 		if ((tmp & ENVY24HT_CCS_I2CSTAT_BSY) == 0)
574 			break;
575 		DELAY(32); /* 31.25kHz */
576 	}
577 	if (i == ENVY24HT_TIMEOUT) {
578 		return -1;
579 	}
580 	envy24ht_wrcs(sc, ENVY24HT_CCS_I2CADDR, addr, 1);
581 	envy24ht_wrcs(sc, ENVY24HT_CCS_I2CDATA, data, 1);
582 	envy24ht_wrcs(sc, ENVY24HT_CCS_I2CDEV,
583 	    (dev & ENVY24HT_CCS_I2CDEV_ADDR) | ENVY24HT_CCS_I2CDEV_WR, 1);
584 	for (i = 0; i < ENVY24HT_TIMEOUT; i++) {
585 		data = envy24ht_rdcs(sc, ENVY24HT_CCS_I2CSTAT, 1);
586 		if ((data & ENVY24HT_CCS_I2CSTAT_BSY) == 0)
587 			break;
588 		DELAY(32); /* 31.25kHz */
589 	}
590 	if (i == ENVY24HT_TIMEOUT) {
591 		return -1;
592 	}
593 
594 	return 0;
595 }
596 
597 static int
598 envy24ht_rdrom(struct sc_info *sc, u_int32_t addr)
599 {
600 	u_int32_t data;
601 
602 #if(0)
603 	device_printf(sc->dev, "envy24ht_rdrom(sc, 0x%02x)\n", addr);
604 #endif
605 	data = envy24ht_rdcs(sc, ENVY24HT_CCS_I2CSTAT, 1);
606 	if ((data & ENVY24HT_CCS_I2CSTAT_ROM) == 0) {
607 #if(0)
608 		device_printf(sc->dev, "envy24ht_rdrom(): E2PROM not presented\n");
609 #endif
610 		return -1;
611 	}
612 
613 	return envy24ht_rdi2c(sc, ENVY24HT_CCS_I2CDEV_ROM, addr);
614 }
615 
616 static struct cfg_info *
617 envy24ht_rom2cfg(struct sc_info *sc)
618 {
619 	struct cfg_info *buff;
620 	int size;
621 	int i;
622 
623 #if(0)
624 	device_printf(sc->dev, "envy24ht_rom2cfg(sc)\n");
625 #endif
626 	size = envy24ht_rdrom(sc, ENVY24HT_E2PROM_SIZE);
627 	if ((size < ENVY24HT_E2PROM_GPIOSTATE + 3) || (size == 0x78)) {
628 #if(0)
629 		device_printf(sc->dev, "envy24ht_rom2cfg(): ENVY24HT_E2PROM_SIZE-->%d\n", size);
630 #endif
631         buff = malloc(sizeof(*buff), M_ENVY24HT, M_NOWAIT);
632         if (buff == NULL) {
633 #if(0)
634                 device_printf(sc->dev, "envy24ht_rom2cfg(): malloc()\n");
635 #endif
636                 return NULL;
637         }
638         buff->free = 1;
639 
640 	/* no valid e2prom, using default values */
641         buff->subvendor = envy24ht_rdrom(sc, ENVY24HT_E2PROM_SUBVENDOR) << 8;
642         buff->subvendor += envy24ht_rdrom(sc, ENVY24HT_E2PROM_SUBVENDOR + 1);
643         buff->subdevice = envy24ht_rdrom(sc, ENVY24HT_E2PROM_SUBDEVICE) << 8;
644         buff->subdevice += envy24ht_rdrom(sc, ENVY24HT_E2PROM_SUBDEVICE + 1);
645         buff->scfg = 0x0b;
646         buff->acl = 0x80;
647         buff->i2s = 0xfc;
648         buff->spdif = 0xc3;
649         buff->gpiomask = 0x21efff;
650         buff->gpiostate = 0x7fffff;
651         buff->gpiodir = 0x5e1000;
652 	buff->cdti = 0x40000;
653 	buff->cclk = 0x80000;
654 	buff->cs = 0x1000;
655 	buff->cif = 0x00;
656 	buff->type = 0x02;
657 
658         for (i = 0; cfg_table[i].subvendor != 0 || cfg_table[i].subdevice != 0;
659 i++)
660                 if (cfg_table[i].subvendor == buff->subvendor &&
661                     cfg_table[i].subdevice == buff->subdevice)
662                         break;
663         buff->name = cfg_table[i].name;
664         buff->codec = cfg_table[i].codec;
665 
666 		return buff;
667 #if 0
668 		return NULL;
669 #endif
670 	}
671 	buff = malloc(sizeof(*buff), M_ENVY24HT, M_NOWAIT);
672 	if (buff == NULL) {
673 #if(0)
674 		device_printf(sc->dev, "envy24ht_rom2cfg(): malloc()\n");
675 #endif
676 		return NULL;
677 	}
678 	buff->free = 1;
679 
680 	buff->subvendor = envy24ht_rdrom(sc, ENVY24HT_E2PROM_SUBVENDOR) << 8;
681 	buff->subvendor += envy24ht_rdrom(sc, ENVY24HT_E2PROM_SUBVENDOR + 1);
682 	buff->subdevice = envy24ht_rdrom(sc, ENVY24HT_E2PROM_SUBDEVICE) << 8;
683 	buff->subdevice += envy24ht_rdrom(sc, ENVY24HT_E2PROM_SUBDEVICE + 1);
684 	buff->scfg = envy24ht_rdrom(sc, ENVY24HT_E2PROM_SCFG);
685 	buff->acl = envy24ht_rdrom(sc, ENVY24HT_E2PROM_ACL);
686 	buff->i2s = envy24ht_rdrom(sc, ENVY24HT_E2PROM_I2S);
687 	buff->spdif = envy24ht_rdrom(sc, ENVY24HT_E2PROM_SPDIF);
688 	buff->gpiomask = envy24ht_rdrom(sc, ENVY24HT_E2PROM_GPIOMASK) | \
689 	envy24ht_rdrom(sc, ENVY24HT_E2PROM_GPIOMASK + 1) << 8 | \
690 	envy24ht_rdrom(sc, ENVY24HT_E2PROM_GPIOMASK + 2) << 16;
691 	buff->gpiostate = envy24ht_rdrom(sc, ENVY24HT_E2PROM_GPIOSTATE) | \
692 	envy24ht_rdrom(sc, ENVY24HT_E2PROM_GPIOSTATE + 1) << 8 | \
693 	envy24ht_rdrom(sc, ENVY24HT_E2PROM_GPIOSTATE + 2) << 16;
694 	buff->gpiodir = envy24ht_rdrom(sc, ENVY24HT_E2PROM_GPIODIR) | \
695 	envy24ht_rdrom(sc, ENVY24HT_E2PROM_GPIODIR + 1) << 8 | \
696 	envy24ht_rdrom(sc, ENVY24HT_E2PROM_GPIODIR + 2) << 16;
697 
698 	for (i = 0; cfg_table[i].subvendor != 0 || cfg_table[i].subdevice != 0; i++)
699 		if (cfg_table[i].subvendor == buff->subvendor &&
700 		    cfg_table[i].subdevice == buff->subdevice)
701 			break;
702 	buff->name = cfg_table[i].name;
703 	buff->codec = cfg_table[i].codec;
704 
705 	return buff;
706 }
707 
708 static void
709 envy24ht_cfgfree(struct cfg_info *cfg) {
710 	if (cfg == NULL)
711 		return;
712 	if (cfg->free)
713 		free(cfg, M_ENVY24HT);
714 	return;
715 }
716 
717 /* -------------------------------------------------------------------- */
718 
719 /* AC'97 codec access routines */
720 
721 #if 0
722 static int
723 envy24ht_coldcd(struct sc_info *sc)
724 {
725 	u_int32_t data;
726 	int i;
727 
728 #if(0)
729 	device_printf(sc->dev, "envy24ht_coldcd()\n");
730 #endif
731 	envy24ht_wrmt(sc, ENVY24HT_MT_AC97CMD, ENVY24HT_MT_AC97CMD_CLD, 1);
732 	DELAY(10);
733 	envy24ht_wrmt(sc, ENVY24HT_MT_AC97CMD, 0, 1);
734 	DELAY(1000);
735 	for (i = 0; i < ENVY24HT_TIMEOUT; i++) {
736 		data = envy24ht_rdmt(sc, ENVY24HT_MT_AC97CMD, 1);
737 		if (data & ENVY24HT_MT_AC97CMD_RDY) {
738 			return 0;
739 		}
740 	}
741 
742 	return -1;
743 }
744 
745 static int
746 envy24ht_slavecd(struct sc_info *sc)
747 {
748 	u_int32_t data;
749 	int i;
750 
751 #if(0)
752 	device_printf(sc->dev, "envy24ht_slavecd()\n");
753 #endif
754 	envy24ht_wrmt(sc, ENVY24HT_MT_AC97CMD,
755 	    ENVY24HT_MT_AC97CMD_CLD | ENVY24HT_MT_AC97CMD_WRM, 1);
756 	DELAY(10);
757 	envy24ht_wrmt(sc, ENVY24HT_MT_AC97CMD, 0, 1);
758 	DELAY(1000);
759 	for (i = 0; i < ENVY24HT_TIMEOUT; i++) {
760 		data = envy24ht_rdmt(sc, ENVY24HT_MT_AC97CMD, 1);
761 		if (data & ENVY24HT_MT_AC97CMD_RDY) {
762 			return 0;
763 		}
764 	}
765 
766 	return -1;
767 }
768 
769 static int
770 envy24ht_rdcd(kobj_t obj, void *devinfo, int regno)
771 {
772 	struct sc_info *sc = (struct sc_info *)devinfo;
773 	u_int32_t data;
774 	int i;
775 
776 #if(0)
777 	device_printf(sc->dev, "envy24ht_rdcd(obj, sc, 0x%02x)\n", regno);
778 #endif
779 	envy24ht_wrmt(sc, ENVY24HT_MT_AC97IDX, (u_int32_t)regno, 1);
780 	envy24ht_wrmt(sc, ENVY24HT_MT_AC97CMD, ENVY24HT_MT_AC97CMD_RD, 1);
781 	for (i = 0; i < ENVY24HT_TIMEOUT; i++) {
782 		data = envy24ht_rdmt(sc, ENVY24HT_MT_AC97CMD, 1);
783 		if ((data & ENVY24HT_MT_AC97CMD_RD) == 0)
784 			break;
785 	}
786 	data = envy24ht_rdmt(sc, ENVY24HT_MT_AC97DLO, 2);
787 
788 #if(0)
789 	device_printf(sc->dev, "envy24ht_rdcd(): return 0x%x\n", data);
790 #endif
791 	return (int)data;
792 }
793 
794 static int
795 envy24ht_wrcd(kobj_t obj, void *devinfo, int regno, u_int16_t data)
796 {
797 	struct sc_info *sc = (struct sc_info *)devinfo;
798 	u_int32_t cmd;
799 	int i;
800 
801 #if(0)
802 	device_printf(sc->dev, "envy24ht_wrcd(obj, sc, 0x%02x, 0x%04x)\n", regno, data);
803 #endif
804 	envy24ht_wrmt(sc, ENVY24HT_MT_AC97IDX, (u_int32_t)regno, 1);
805 	envy24ht_wrmt(sc, ENVY24HT_MT_AC97DLO, (u_int32_t)data, 2);
806 	envy24ht_wrmt(sc, ENVY24HT_MT_AC97CMD, ENVY24HT_MT_AC97CMD_WR, 1);
807 	for (i = 0; i < ENVY24HT_TIMEOUT; i++) {
808 		cmd = envy24ht_rdmt(sc, ENVY24HT_MT_AC97CMD, 1);
809 		if ((cmd & ENVY24HT_MT_AC97CMD_WR) == 0)
810 			break;
811 	}
812 
813 	return 0;
814 }
815 
816 static kobj_method_t envy24ht_ac97_methods[] = {
817 	KOBJMETHOD(ac97_read,	envy24ht_rdcd),
818 	KOBJMETHOD(ac97_write,	envy24ht_wrcd),
819 	KOBJMETHOD_END
820 };
821 AC97_DECLARE(envy24ht_ac97);
822 #endif
823 
824 /* -------------------------------------------------------------------- */
825 
826 /* GPIO access routines */
827 
828 static u_int32_t
829 envy24ht_gpiord(struct sc_info *sc)
830 {
831 	if (sc->cfg->subvendor == 0x153b  && sc->cfg->subdevice == 0x1150)
832 	return envy24ht_rdcs(sc, ENVY24HT_CCS_GPIO_LDATA, 2);
833 	else
834 	return (envy24ht_rdcs(sc, ENVY24HT_CCS_GPIO_HDATA, 1) << 16 | envy24ht_rdcs(sc, ENVY24HT_CCS_GPIO_LDATA, 2));
835 }
836 
837 static void
838 envy24ht_gpiowr(struct sc_info *sc, u_int32_t data)
839 {
840 #if(0)
841 	device_printf(sc->dev, "envy24ht_gpiowr(sc, 0x%02x)\n", data & 0x7FFFFF);
842 	return;
843 #endif
844 	envy24ht_wrcs(sc, ENVY24HT_CCS_GPIO_LDATA, data, 2);
845 	if (sc->cfg->subdevice != 0x1150)
846 	envy24ht_wrcs(sc, ENVY24HT_CCS_GPIO_HDATA, data >> 16, 1);
847 	return;
848 }
849 
850 #if 0
851 static u_int32_t
852 envy24ht_gpiogetmask(struct sc_info *sc)
853 {
854 	return (envy24ht_rdcs(sc, ENVY24HT_CCS_GPIO_HMASK, 1) << 16 | envy24ht_rdcs(sc, ENVY24HT_CCS_GPIO_LMASK, 2));
855 }
856 #endif
857 
858 static void
859 envy24ht_gpiosetmask(struct sc_info *sc, u_int32_t mask)
860 {
861         envy24ht_wrcs(sc, ENVY24HT_CCS_GPIO_LMASK, mask, 2);
862 	if (sc->cfg->subdevice != 0x1150)
863         envy24ht_wrcs(sc, ENVY24HT_CCS_GPIO_HMASK, mask >> 16, 1);
864 	return;
865 }
866 
867 #if 0
868 static u_int32_t
869 envy24ht_gpiogetdir(struct sc_info *sc)
870 {
871 	return envy24ht_rdcs(sc, ENVY24HT_CCS_GPIO_CTLDIR, 4);
872 }
873 #endif
874 
875 static void
876 envy24ht_gpiosetdir(struct sc_info *sc, u_int32_t dir)
877 {
878 	if (sc->cfg->subvendor == 0x153b  && sc->cfg->subdevice == 0x1150)
879 	envy24ht_wrcs(sc, ENVY24HT_CCS_GPIO_CTLDIR, dir, 2);
880 	else
881 	envy24ht_wrcs(sc, ENVY24HT_CCS_GPIO_CTLDIR, dir, 4);
882 	return;
883 }
884 
885 /* -------------------------------------------------------------------- */
886 
887 /* SPI codec access interface routine */
888 
889 struct envy24ht_spi_codec {
890 	struct spicds_info *info;
891 	struct sc_info *parent;
892 	int dir;
893 	int num;
894 	int cs, cclk, cdti;
895 };
896 
897 static void
898 envy24ht_spi_ctl(void *codec, unsigned int cs, unsigned int cclk, unsigned int cdti)
899 {
900 	u_int32_t data = 0;
901 	struct envy24ht_spi_codec *ptr = codec;
902 
903 #if(0)
904 	device_printf(ptr->parent->dev, "--> %d, %d, %d\n", cs, cclk, cdti);
905 #endif
906 	data = envy24ht_gpiord(ptr->parent);
907 	data &= ~(ptr->cs | ptr->cclk | ptr->cdti);
908 	if (cs) data += ptr->cs;
909 	if (cclk) data += ptr->cclk;
910 	if (cdti) data += ptr->cdti;
911 	envy24ht_gpiowr(ptr->parent, data);
912 	return;
913 }
914 
915 static void *
916 envy24ht_spi_create(device_t dev, void *info, int dir, int num)
917 {
918 	struct sc_info *sc = info;
919 	struct envy24ht_spi_codec *buff = NULL;
920 
921 #if(0)
922 	device_printf(sc->dev, "envy24ht_spi_create(dev, sc, %d, %d)\n", dir, num);
923 #endif
924 
925 	buff = malloc(sizeof(*buff), M_ENVY24HT, M_NOWAIT);
926 	if (buff == NULL)
927 		return NULL;
928 
929 	if (dir == PCMDIR_REC && sc->adc[num] != NULL)
930 		buff->info = ((struct envy24ht_spi_codec *)sc->adc[num])->info;
931 	else if (dir == PCMDIR_PLAY && sc->dac[num] != NULL)
932 		buff->info = ((struct envy24ht_spi_codec *)sc->dac[num])->info;
933 	else
934 		buff->info = spicds_create(dev, buff, num, envy24ht_spi_ctl);
935 	if (buff->info == NULL) {
936 		free(buff, M_ENVY24HT);
937 		return NULL;
938 	}
939 
940 	buff->parent = sc;
941 	buff->dir = dir;
942 	buff->num = num;
943 
944 	return (void *)buff;
945 }
946 
947 static void
948 envy24ht_spi_destroy(void *codec)
949 {
950 	struct envy24ht_spi_codec *ptr = codec;
951 	if (ptr == NULL)
952 		return;
953 #if(0)
954 	device_printf(ptr->parent->dev, "envy24ht_spi_destroy()\n");
955 #endif
956 
957 	if (ptr->dir == PCMDIR_PLAY) {
958 		if (ptr->parent->dac[ptr->num] != NULL)
959 			spicds_destroy(ptr->info);
960 	}
961 	else {
962 		if (ptr->parent->adc[ptr->num] != NULL)
963 			spicds_destroy(ptr->info);
964 	}
965 
966 	free(codec, M_ENVY24HT);
967 }
968 
969 static void
970 envy24ht_spi_init(void *codec)
971 {
972 	struct envy24ht_spi_codec *ptr = codec;
973 	if (ptr == NULL)
974 		return;
975 #if(0)
976 	device_printf(ptr->parent->dev, "envy24ht_spicds_init()\n");
977 #endif
978         ptr->cs = ptr->parent->cfg->cs;
979 	ptr->cclk = ptr->parent->cfg->cclk;
980 	ptr->cdti =  ptr->parent->cfg->cdti;
981 	spicds_settype(ptr->info, ptr->parent->cfg->type);
982 	spicds_setcif(ptr->info, ptr->parent->cfg->cif);
983 	if (ptr->parent->cfg->type == SPICDS_TYPE_AK4524 || \
984 	ptr->parent->cfg->type == SPICDS_TYPE_AK4528) {
985 	spicds_setformat(ptr->info,
986 	    AK452X_FORMAT_I2S | AK452X_FORMAT_256FSN | AK452X_FORMAT_1X);
987 	spicds_setdvc(ptr->info, AK452X_DVC_DEMOFF);
988 	}
989 
990 	/* for the time being, init only first codec */
991 	if (ptr->num == 0)
992 	spicds_init(ptr->info);
993 }
994 
995 static void
996 envy24ht_spi_reinit(void *codec)
997 {
998 	struct envy24ht_spi_codec *ptr = codec;
999 	if (ptr == NULL)
1000 		return;
1001 #if(0)
1002 	device_printf(ptr->parent->dev, "envy24ht_spi_reinit()\n");
1003 #endif
1004 
1005 	spicds_reinit(ptr->info);
1006 }
1007 
1008 static void
1009 envy24ht_spi_setvolume(void *codec, int dir, unsigned int left, unsigned int right)
1010 {
1011 	struct envy24ht_spi_codec *ptr = codec;
1012 	if (ptr == NULL)
1013 		return;
1014 #if(0)
1015 	device_printf(ptr->parent->dev, "envy24ht_spi_set()\n");
1016 #endif
1017 
1018 	spicds_set(ptr->info, dir, left, right);
1019 }
1020 
1021 /* -------------------------------------------------------------------- */
1022 
1023 /* hardware access routeines */
1024 
1025 static struct {
1026 	u_int32_t speed;
1027 	u_int32_t code;
1028 } envy24ht_speedtab[] = {
1029 	{48000, ENVY24HT_MT_RATE_48000},
1030 	{24000, ENVY24HT_MT_RATE_24000},
1031 	{12000, ENVY24HT_MT_RATE_12000},
1032 	{9600, ENVY24HT_MT_RATE_9600},
1033 	{32000, ENVY24HT_MT_RATE_32000},
1034 	{16000, ENVY24HT_MT_RATE_16000},
1035 	{8000, ENVY24HT_MT_RATE_8000},
1036 	{96000, ENVY24HT_MT_RATE_96000},
1037 	{192000, ENVY24HT_MT_RATE_192000},
1038 	{64000, ENVY24HT_MT_RATE_64000},
1039 	{44100, ENVY24HT_MT_RATE_44100},
1040 	{22050, ENVY24HT_MT_RATE_22050},
1041 	{11025, ENVY24HT_MT_RATE_11025},
1042 	{88200, ENVY24HT_MT_RATE_88200},
1043 	{176400, ENVY24HT_MT_RATE_176400},
1044 	{0, 0x10}
1045 };
1046 
1047 static u_int32_t
1048 envy24ht_setspeed(struct sc_info *sc, u_int32_t speed) {
1049 	u_int32_t code, i2sfmt;
1050 	int i = 0;
1051 
1052 #if(0)
1053 	device_printf(sc->dev, "envy24ht_setspeed(sc, %d)\n", speed);
1054 	if (speed == 0) {
1055 		code = ENVY24HT_MT_RATE_SPDIF; /* external master clock */
1056 		envy24ht_slavecd(sc);
1057 	}
1058 	else {
1059 #endif
1060 		for (i = 0; envy24ht_speedtab[i].speed != 0; i++) {
1061 			if (envy24ht_speedtab[i].speed == speed)
1062 				break;
1063 		}
1064 		code = envy24ht_speedtab[i].code;
1065 #if 0
1066 	}
1067 	device_printf(sc->dev, "envy24ht_setspeed(): speed %d/code 0x%04x\n", envy24ht_speedtab[i].speed, code);
1068 #endif
1069 	if (code < 0x10) {
1070 		envy24ht_wrmt(sc, ENVY24HT_MT_RATE, code, 1);
1071 		if ((((sc->cfg->scfg & ENVY24HT_CCSM_SCFG_XIN2) == 0x00) && (code == ENVY24HT_MT_RATE_192000)) || \
1072 									    (code == ENVY24HT_MT_RATE_176400)) {
1073 			i2sfmt = envy24ht_rdmt(sc, ENVY24HT_MT_I2S, 1);
1074 			i2sfmt |= ENVY24HT_MT_I2S_MLR128;
1075 			envy24ht_wrmt(sc, ENVY24HT_MT_I2S, i2sfmt, 1);
1076 		}
1077 		else {
1078 			i2sfmt = envy24ht_rdmt(sc, ENVY24HT_MT_I2S, 1);
1079 			i2sfmt &= ~ENVY24HT_MT_I2S_MLR128;
1080 			envy24ht_wrmt(sc, ENVY24HT_MT_I2S, i2sfmt, 1);
1081 		}
1082 		code = envy24ht_rdmt(sc, ENVY24HT_MT_RATE, 1);
1083 		code &= ENVY24HT_MT_RATE_MASK;
1084 		for (i = 0; envy24ht_speedtab[i].code < 0x10; i++) {
1085 			if (envy24ht_speedtab[i].code == code)
1086 				break;
1087 		}
1088 		speed = envy24ht_speedtab[i].speed;
1089 	}
1090 	else
1091 		speed = 0;
1092 
1093 #if(0)
1094 	device_printf(sc->dev, "envy24ht_setspeed(): return %d\n", speed);
1095 #endif
1096 	return speed;
1097 }
1098 
1099 static void
1100 envy24ht_setvolume(struct sc_info *sc, unsigned ch)
1101 {
1102 #if(0)
1103 	device_printf(sc->dev, "envy24ht_setvolume(sc, %d)\n", ch);
1104 	envy24ht_wrmt(sc, ENVY24HT_MT_VOLIDX, ch * 2, 1);
1105 	envy24ht_wrmt(sc, ENVY24HT_MT_VOLUME, 0x7f00 | sc->left[ch], 2);
1106 	envy24ht_wrmt(sc, ENVY24HT_MT_VOLIDX, ch * 2 + 1, 1);
1107 	envy24ht_wrmt(sc, ENVY24HT_MT_VOLUME, (sc->right[ch] << 8) | 0x7f, 2);
1108 #endif
1109 }
1110 
1111 static void
1112 envy24ht_mutevolume(struct sc_info *sc, unsigned ch)
1113 {
1114 #if 0
1115 	u_int32_t vol;
1116 
1117 	device_printf(sc->dev, "envy24ht_mutevolume(sc, %d)\n", ch);
1118 	vol = ENVY24HT_VOL_MUTE << 8 | ENVY24HT_VOL_MUTE;
1119 	envy24ht_wrmt(sc, ENVY24HT_MT_VOLIDX, ch * 2, 1);
1120 	envy24ht_wrmt(sc, ENVY24HT_MT_VOLUME, vol, 2);
1121 	envy24ht_wrmt(sc, ENVY24HT_MT_VOLIDX, ch * 2 + 1, 1);
1122 	envy24ht_wrmt(sc, ENVY24HT_MT_VOLUME, vol, 2);
1123 #endif
1124 }
1125 
1126 static u_int32_t
1127 envy24ht_gethwptr(struct sc_info *sc, int dir)
1128 {
1129 	int unit, regno;
1130 	u_int32_t ptr, rtn;
1131 
1132 #if(0)
1133 	device_printf(sc->dev, "envy24ht_gethwptr(sc, %d)\n", dir);
1134 #endif
1135 	if (dir == PCMDIR_PLAY) {
1136 		rtn = sc->psize / 4;
1137 		unit = ENVY24HT_PLAY_BUFUNIT / 4;
1138 		regno = ENVY24HT_MT_PCNT;
1139 	}
1140 	else {
1141 		rtn = sc->rsize / 4;
1142 		unit = ENVY24HT_REC_BUFUNIT / 4;
1143 		regno = ENVY24HT_MT_RCNT;
1144 	}
1145 
1146 	ptr = envy24ht_rdmt(sc, regno, 2);
1147 	rtn -= (ptr + 1);
1148 	rtn /= unit;
1149 
1150 #if(0)
1151 	device_printf(sc->dev, "envy24ht_gethwptr(): return %d\n", rtn);
1152 #endif
1153 	return rtn;
1154 }
1155 
1156 static void
1157 envy24ht_updintr(struct sc_info *sc, int dir)
1158 {
1159 	int regintr;
1160 	u_int32_t mask, intr;
1161 	u_int32_t cnt;
1162 	u_int16_t blk;
1163 
1164 #if(0)
1165 	device_printf(sc->dev, "envy24ht_updintr(sc, %d)\n", dir);
1166 #endif
1167 	if (dir == PCMDIR_PLAY) {
1168 		blk = sc->blk[0];
1169 		regintr = ENVY24HT_MT_PTERM;
1170 		mask = ~ENVY24HT_MT_INT_PMASK;
1171 	}
1172 	else {
1173 		blk = sc->blk[1];
1174 		regintr = ENVY24HT_MT_RTERM;
1175 		mask = ~ENVY24HT_MT_INT_RMASK;
1176 	}
1177 
1178 	cnt = blk - 1;
1179 #if(0)
1180 	device_printf(sc->dev, "envy24ht_updintr():blk = %d, cnt = %d\n", blk, cnt);
1181 #endif
1182 	envy24ht_wrmt(sc, regintr, cnt, 2);
1183 	intr = envy24ht_rdmt(sc, ENVY24HT_MT_INT_MASK, 1);
1184 #if(0)
1185 	device_printf(sc->dev, "envy24ht_updintr():intr = 0x%02x, mask = 0x%02x\n", intr, mask);
1186 #endif
1187 	envy24ht_wrmt(sc, ENVY24HT_MT_INT_MASK, intr & mask, 1);
1188 #if(0)
1189 	device_printf(sc->dev, "envy24ht_updintr():INT-->0x%02x\n",
1190 		      envy24ht_rdmt(sc, ENVY24HT_MT_INT_MASK, 1));
1191 #endif
1192 
1193 	return;
1194 }
1195 
1196 #if 0
1197 static void
1198 envy24ht_maskintr(struct sc_info *sc, int dir)
1199 {
1200 	u_int32_t mask, intr;
1201 
1202 #if(0)
1203 	device_printf(sc->dev, "envy24ht_maskintr(sc, %d)\n", dir);
1204 #endif
1205 	if (dir == PCMDIR_PLAY)
1206 		mask = ENVY24HT_MT_INT_PMASK;
1207 	else
1208 		mask = ENVY24HT_MT_INT_RMASK;
1209 	intr = envy24ht_rdmt(sc, ENVY24HT_MT_INT, 1);
1210 	envy24ht_wrmt(sc, ENVY24HT_MT_INT, intr | mask, 1);
1211 
1212 	return;
1213 }
1214 #endif
1215 
1216 static int
1217 envy24ht_checkintr(struct sc_info *sc, int dir)
1218 {
1219 	u_int32_t mask, stat, intr, rtn;
1220 
1221 #if(0)
1222 	device_printf(sc->dev, "envy24ht_checkintr(sc, %d)\n", dir);
1223 #endif
1224 	intr = envy24ht_rdmt(sc, ENVY24HT_MT_INT_STAT, 1);
1225 	if (dir == PCMDIR_PLAY) {
1226 		if ((rtn = intr & ENVY24HT_MT_INT_PSTAT) != 0) {
1227 			mask = ~ENVY24HT_MT_INT_RSTAT;
1228 			envy24ht_wrmt(sc, 0x1a, 0x01, 1);
1229 			envy24ht_wrmt(sc, ENVY24HT_MT_INT_STAT, (intr & mask) | ENVY24HT_MT_INT_PSTAT | 0x08, 1);
1230 			stat = envy24ht_rdmt(sc, ENVY24HT_MT_INT_MASK, 1);
1231 			envy24ht_wrmt(sc, ENVY24HT_MT_INT_MASK, stat | ENVY24HT_MT_INT_PMASK, 1);
1232 		}
1233 	}
1234 	else {
1235 		if ((rtn = intr & ENVY24HT_MT_INT_RSTAT) != 0) {
1236 			mask = ~ENVY24HT_MT_INT_PSTAT;
1237 #if 0
1238 			stat = ENVY24HT_MT_INT_RSTAT | ENVY24HT_MT_INT_RMASK;
1239 #endif
1240 			envy24ht_wrmt(sc, ENVY24HT_MT_INT_STAT, (intr & mask) | ENVY24HT_MT_INT_RSTAT, 1);
1241 			stat = envy24ht_rdmt(sc, ENVY24HT_MT_INT_MASK, 1);
1242 			envy24ht_wrmt(sc, ENVY24HT_MT_INT_MASK, stat | ENVY24HT_MT_INT_RMASK, 1);
1243 		}
1244 	}
1245 
1246 	return rtn;
1247 }
1248 
1249 static void
1250 envy24ht_start(struct sc_info *sc, int dir)
1251 {
1252 	u_int32_t stat, sw;
1253 
1254 #if(0)
1255 	device_printf(sc->dev, "envy24ht_start(sc, %d)\n", dir);
1256 #endif
1257 	if (dir == PCMDIR_PLAY)
1258 		sw = ENVY24HT_MT_PCTL_PSTART;
1259 	else
1260 		sw = ENVY24HT_MT_PCTL_RSTART;
1261 
1262 	stat = envy24ht_rdmt(sc, ENVY24HT_MT_PCTL, 1);
1263 	envy24ht_wrmt(sc, ENVY24HT_MT_PCTL, stat | sw, 1);
1264 #if(0)
1265 	DELAY(100);
1266 	device_printf(sc->dev, "PADDR:0x%08x\n", envy24ht_rdmt(sc, ENVY24HT_MT_PADDR, 4));
1267 	device_printf(sc->dev, "PCNT:%ld\n", envy24ht_rdmt(sc, ENVY24HT_MT_PCNT, 2));
1268 #endif
1269 
1270 	return;
1271 }
1272 
1273 static void
1274 envy24ht_stop(struct sc_info *sc, int dir)
1275 {
1276 	u_int32_t stat, sw;
1277 
1278 #if(0)
1279 	device_printf(sc->dev, "envy24ht_stop(sc, %d)\n", dir);
1280 #endif
1281 	if (dir == PCMDIR_PLAY)
1282 		sw = ~ENVY24HT_MT_PCTL_PSTART;
1283 	else
1284 		sw = ~ENVY24HT_MT_PCTL_RSTART;
1285 
1286 	stat = envy24ht_rdmt(sc, ENVY24HT_MT_PCTL, 1);
1287 	envy24ht_wrmt(sc, ENVY24HT_MT_PCTL, stat & sw, 1);
1288 
1289 	return;
1290 }
1291 
1292 #if 0
1293 static int
1294 envy24ht_route(struct sc_info *sc, int dac, int class, int adc, int rev)
1295 {
1296 	return 0;
1297 }
1298 #endif
1299 
1300 /* -------------------------------------------------------------------- */
1301 
1302 /* buffer copy routines */
1303 static void
1304 envy24ht_p32sl(struct sc_chinfo *ch)
1305 {
1306 	int length;
1307 	sample32_t *dmabuf;
1308 	u_int32_t *data;
1309 	int src, dst, ssize, dsize, slot;
1310 	int i;
1311 
1312 	length = sndbuf_getready(ch->buffer) / 8;
1313 	dmabuf = ch->parent->pbuf;
1314 	data = (u_int32_t *)ch->data;
1315 	src = sndbuf_getreadyptr(ch->buffer) / 4;
1316 	dst = src / 2 + ch->offset;
1317 	ssize = ch->size / 4;
1318 	dsize = ch->size / 8;
1319 	slot = ch->num * 2;
1320 
1321 	for (i = 0; i < length; i++) {
1322 		dmabuf[dst * ENVY24HT_PLAY_CHNUM + slot].buffer = data[src];
1323 		dmabuf[dst * ENVY24HT_PLAY_CHNUM + slot + 1].buffer = data[src + 1];
1324 		dst++;
1325 		dst %= dsize;
1326 		src += 2;
1327 		src %= ssize;
1328 	}
1329 
1330 	return;
1331 }
1332 
1333 static void
1334 envy24ht_p16sl(struct sc_chinfo *ch)
1335 {
1336 	int length;
1337 	sample32_t *dmabuf;
1338 	u_int16_t *data;
1339 	int src, dst, ssize, dsize, slot;
1340 	int i;
1341 
1342 #if(0)
1343 	device_printf(ch->parent->dev, "envy24ht_p16sl()\n");
1344 #endif
1345 	length = sndbuf_getready(ch->buffer) / 4;
1346 	dmabuf = ch->parent->pbuf;
1347 	data = (u_int16_t *)ch->data;
1348 	src = sndbuf_getreadyptr(ch->buffer) / 2;
1349 	dst = src / 2 + ch->offset;
1350 	ssize = ch->size / 2;
1351 	dsize = ch->size / 4;
1352 	slot = ch->num * 2;
1353 #if(0)
1354 	device_printf(ch->parent->dev, "envy24ht_p16sl():%lu-->%lu(%lu)\n", src, dst, length);
1355 #endif
1356 
1357 	for (i = 0; i < length; i++) {
1358 		dmabuf[dst * ENVY24HT_PLAY_CHNUM + slot].buffer = (u_int32_t)data[src] << 16;
1359 		dmabuf[dst * ENVY24HT_PLAY_CHNUM + slot + 1].buffer = (u_int32_t)data[src + 1] << 16;
1360 #if(0)
1361 		if (i < 16) {
1362 			printf("%08x", dmabuf[dst * ENVY24HT_PLAY_CHNUM + slot]);
1363 			printf("%08x", dmabuf[dst * ENVY24HT_PLAY_CHNUM + slot + 1]);
1364 		}
1365 #endif
1366 		dst++;
1367 		dst %= dsize;
1368 		src += 2;
1369 		src %= ssize;
1370 	}
1371 #if(0)
1372 	printf("\n");
1373 #endif
1374 
1375 	return;
1376 }
1377 
1378 static void
1379 envy24ht_p8u(struct sc_chinfo *ch)
1380 {
1381 	int length;
1382 	sample32_t *dmabuf;
1383 	u_int8_t *data;
1384 	int src, dst, ssize, dsize, slot;
1385 	int i;
1386 
1387 	length = sndbuf_getready(ch->buffer) / 2;
1388 	dmabuf = ch->parent->pbuf;
1389 	data = (u_int8_t *)ch->data;
1390 	src = sndbuf_getreadyptr(ch->buffer);
1391 	dst = src / 2 + ch->offset;
1392 	ssize = ch->size;
1393 	dsize = ch->size / 4;
1394 	slot = ch->num * 2;
1395 
1396 	for (i = 0; i < length; i++) {
1397 		dmabuf[dst * ENVY24HT_PLAY_CHNUM + slot].buffer = ((u_int32_t)data[src] ^ 0x80) << 24;
1398 		dmabuf[dst * ENVY24HT_PLAY_CHNUM + slot + 1].buffer = ((u_int32_t)data[src + 1] ^ 0x80) << 24;
1399 		dst++;
1400 		dst %= dsize;
1401 		src += 2;
1402 		src %= ssize;
1403 	}
1404 
1405 	return;
1406 }
1407 
1408 static void
1409 envy24ht_r32sl(struct sc_chinfo *ch)
1410 {
1411 	int length;
1412 	sample32_t *dmabuf;
1413 	u_int32_t *data;
1414 	int src, dst, ssize, dsize, slot;
1415 	int i;
1416 
1417 	length = sndbuf_getfree(ch->buffer) / 8;
1418 	dmabuf = ch->parent->rbuf;
1419 	data = (u_int32_t *)ch->data;
1420 	dst = sndbuf_getfreeptr(ch->buffer) / 4;
1421 	src = dst / 2 + ch->offset;
1422 	dsize = ch->size / 4;
1423 	ssize = ch->size / 8;
1424 	slot = (ch->num - ENVY24HT_CHAN_REC_ADC1) * 2;
1425 
1426 	for (i = 0; i < length; i++) {
1427 		data[dst] = dmabuf[src * ENVY24HT_REC_CHNUM + slot].buffer;
1428 		data[dst + 1] = dmabuf[src * ENVY24HT_REC_CHNUM + slot + 1].buffer;
1429 		dst += 2;
1430 		dst %= dsize;
1431 		src++;
1432 		src %= ssize;
1433 	}
1434 
1435 	return;
1436 }
1437 
1438 static void
1439 envy24ht_r16sl(struct sc_chinfo *ch)
1440 {
1441 	int length;
1442 	sample32_t *dmabuf;
1443 	u_int16_t *data;
1444 	int src, dst, ssize, dsize, slot;
1445 	int i;
1446 
1447 	length = sndbuf_getfree(ch->buffer) / 4;
1448 	dmabuf = ch->parent->rbuf;
1449 	data = (u_int16_t *)ch->data;
1450 	dst = sndbuf_getfreeptr(ch->buffer) / 2;
1451 	src = dst / 2 + ch->offset;
1452 	dsize = ch->size / 2;
1453 	ssize = ch->size / 8;
1454 	slot = (ch->num - ENVY24HT_CHAN_REC_ADC1) * 2;
1455 
1456 	for (i = 0; i < length; i++) {
1457 		data[dst] = dmabuf[src * ENVY24HT_REC_CHNUM + slot].buffer;
1458 		data[dst + 1] = dmabuf[src * ENVY24HT_REC_CHNUM + slot + 1].buffer;
1459 		dst += 2;
1460 		dst %= dsize;
1461 		src++;
1462 		src %= ssize;
1463 	}
1464 
1465 	return;
1466 }
1467 
1468 /* -------------------------------------------------------------------- */
1469 
1470 /* channel interface */
1471 static void *
1472 envy24htchan_init(kobj_t obj, void *devinfo, struct snd_dbuf *b, struct pcm_channel *c, int dir)
1473 {
1474 	struct sc_info	*sc = (struct sc_info *)devinfo;
1475 	struct sc_chinfo *ch;
1476 	unsigned num;
1477 
1478 #if(0)
1479 	device_printf(sc->dev, "envy24htchan_init(obj, devinfo, b, c, %d)\n", dir);
1480 #endif
1481 	snd_mtxlock(sc->lock);
1482 #if 0
1483 	if ((sc->chnum > ENVY24HT_CHAN_PLAY_SPDIF && dir != PCMDIR_REC) ||
1484 	    (sc->chnum < ENVY24HT_CHAN_REC_ADC1 && dir != PCMDIR_PLAY)) {
1485 		snd_mtxunlock(sc->lock);
1486 		return NULL;
1487 	}
1488 #endif
1489 	num = sc->chnum;
1490 
1491 	ch = &sc->chan[num];
1492 	ch->size = 8 * ENVY24HT_SAMPLE_NUM;
1493 	ch->data = malloc(ch->size, M_ENVY24HT, M_NOWAIT);
1494 	if (ch->data == NULL) {
1495 		ch->size = 0;
1496 		ch = NULL;
1497 	}
1498 	else {
1499 		ch->buffer = b;
1500 		ch->channel = c;
1501 		ch->parent = sc;
1502 		ch->dir = dir;
1503 		/* set channel map */
1504 		ch->num = envy24ht_chanmap[num];
1505 		snd_mtxunlock(sc->lock);
1506 		sndbuf_setup(ch->buffer, ch->data, ch->size);
1507 		snd_mtxlock(sc->lock);
1508 		/* these 2 values are dummy */
1509 		ch->unit = 4;
1510 		ch->blk = 10240;
1511 	}
1512 	snd_mtxunlock(sc->lock);
1513 
1514 	return ch;
1515 }
1516 
1517 static int
1518 envy24htchan_free(kobj_t obj, void *data)
1519 {
1520 	struct sc_chinfo *ch = data;
1521 	struct sc_info *sc = ch->parent;
1522 
1523 #if(0)
1524 	device_printf(sc->dev, "envy24htchan_free()\n");
1525 #endif
1526 	snd_mtxlock(sc->lock);
1527 	if (ch->data != NULL) {
1528 		free(ch->data, M_ENVY24HT);
1529 		ch->data = NULL;
1530 	}
1531 	snd_mtxunlock(sc->lock);
1532 
1533 	return 0;
1534 }
1535 
1536 static int
1537 envy24htchan_setformat(kobj_t obj, void *data, u_int32_t format)
1538 {
1539 	struct sc_chinfo *ch = data;
1540 	struct sc_info *sc = ch->parent;
1541 	struct envy24ht_emldma *emltab;
1542 	/* unsigned int bcnt, bsize; */
1543 	int i;
1544 
1545 #if(0)
1546 	device_printf(sc->dev, "envy24htchan_setformat(obj, data, 0x%08x)\n", format);
1547 #endif
1548 	snd_mtxlock(sc->lock);
1549 	/* check and get format related information */
1550 	if (ch->dir == PCMDIR_PLAY)
1551 		emltab = envy24ht_pemltab;
1552 	else
1553 		emltab = envy24ht_remltab;
1554 	if (emltab == NULL) {
1555 		snd_mtxunlock(sc->lock);
1556 		return -1;
1557 	}
1558 	for (i = 0; emltab[i].format != 0; i++)
1559 		if (emltab[i].format == format)
1560 			break;
1561 	if (emltab[i].format == 0) {
1562 		snd_mtxunlock(sc->lock);
1563 		return -1;
1564 	}
1565 
1566 	/* set format information */
1567 	ch->format = format;
1568 	ch->emldma = emltab[i].emldma;
1569 	if (ch->unit > emltab[i].unit)
1570 		ch->blk *= ch->unit / emltab[i].unit;
1571 	else
1572 		ch->blk /= emltab[i].unit / ch->unit;
1573 	ch->unit = emltab[i].unit;
1574 
1575 	/* set channel buffer information */
1576 	ch->size = ch->unit * ENVY24HT_SAMPLE_NUM;
1577 #if 0
1578 	if (ch->dir == PCMDIR_PLAY)
1579 		bsize = ch->blk * 4 / ENVY24HT_PLAY_BUFUNIT;
1580 	else
1581 		bsize = ch->blk * 4 / ENVY24HT_REC_BUFUNIT;
1582 	bsize *= ch->unit;
1583 	bcnt = ch->size / bsize;
1584 	sndbuf_resize(ch->buffer, bcnt, bsize);
1585 #endif
1586 	snd_mtxunlock(sc->lock);
1587 
1588 #if(0)
1589 	device_printf(sc->dev, "envy24htchan_setformat(): return 0x%08x\n", 0);
1590 #endif
1591 	return 0;
1592 }
1593 
1594 /*
1595   IMPLEMENT NOTICE: In this driver, setspeed function only do setting
1596   of speed information value. And real hardware speed setting is done
1597   at start triggered(see envy24htchan_trigger()). So, at this function
1598   is called, any value that ENVY24 can use is able to set. But, at
1599   start triggerd, some other channel is running, and that channel's
1600   speed isn't same with, then trigger function will fail.
1601 */
1602 static u_int32_t
1603 envy24htchan_setspeed(kobj_t obj, void *data, u_int32_t speed)
1604 {
1605 	struct sc_chinfo *ch = data;
1606 	u_int32_t val, prev;
1607 	int i;
1608 
1609 #if(0)
1610 	device_printf(ch->parent->dev, "envy24htchan_setspeed(obj, data, %d)\n", speed);
1611 #endif
1612 	prev = 0x7fffffff;
1613 	for (i = 0; (val = envy24ht_speed[i]) != 0; i++) {
1614 		if (abs(val - speed) < abs(prev - speed))
1615 			prev = val;
1616 		else
1617 			break;
1618 	}
1619 	ch->speed = prev;
1620 
1621 #if(0)
1622 	device_printf(ch->parent->dev, "envy24htchan_setspeed(): return %d\n", ch->speed);
1623 #endif
1624 	return ch->speed;
1625 }
1626 
1627 static u_int32_t
1628 envy24htchan_setblocksize(kobj_t obj, void *data, u_int32_t blocksize)
1629 {
1630 	struct sc_chinfo *ch = data;
1631 	/* struct sc_info *sc = ch->parent; */
1632 	u_int32_t size, prev;
1633 	unsigned int bcnt, bsize;
1634 
1635 #if(0)
1636 	device_printf(sc->dev, "envy24htchan_setblocksize(obj, data, %d)\n", blocksize);
1637 #endif
1638 	prev = 0x7fffffff;
1639 	/* snd_mtxlock(sc->lock); */
1640 	for (size = ch->size / 2; size > 0; size /= 2) {
1641 		if (abs(size - blocksize) < abs(prev - blocksize))
1642 			prev = size;
1643 		else
1644 			break;
1645 	}
1646 
1647 	ch->blk = prev / ch->unit;
1648 	if (ch->dir == PCMDIR_PLAY)
1649 		ch->blk *= ENVY24HT_PLAY_BUFUNIT / 4;
1650 	else
1651 		ch->blk *= ENVY24HT_REC_BUFUNIT / 4;
1652         /* set channel buffer information */
1653         /* ch->size = ch->unit * ENVY24HT_SAMPLE_NUM; */
1654         if (ch->dir == PCMDIR_PLAY)
1655                 bsize = ch->blk * 4 / ENVY24HT_PLAY_BUFUNIT;
1656         else
1657                 bsize = ch->blk * 4 / ENVY24HT_REC_BUFUNIT;
1658         bsize *= ch->unit;
1659         bcnt = ch->size / bsize;
1660         sndbuf_resize(ch->buffer, bcnt, bsize);
1661 	/* snd_mtxunlock(sc->lock); */
1662 
1663 #if(0)
1664 	device_printf(sc->dev, "envy24htchan_setblocksize(): return %d\n", prev);
1665 #endif
1666 	return prev;
1667 }
1668 
1669 /* semantic note: must start at beginning of buffer */
1670 static int
1671 envy24htchan_trigger(kobj_t obj, void *data, int go)
1672 {
1673 	struct sc_chinfo *ch = data;
1674 	struct sc_info *sc = ch->parent;
1675 	u_int32_t ptr;
1676 	int slot;
1677 	int error = 0;
1678 #if 0
1679 	int i;
1680 
1681 	device_printf(sc->dev, "envy24htchan_trigger(obj, data, %d)\n", go);
1682 #endif
1683 	snd_mtxlock(sc->lock);
1684 	if (ch->dir == PCMDIR_PLAY)
1685 		slot = 0;
1686 	else
1687 		slot = 1;
1688 	switch (go) {
1689 	case PCMTRIG_START:
1690 #if(0)
1691 		device_printf(sc->dev, "envy24htchan_trigger(): start\n");
1692 #endif
1693 		/* check or set channel speed */
1694 		if (sc->run[0] == 0 && sc->run[1] == 0) {
1695 			sc->speed = envy24ht_setspeed(sc, ch->speed);
1696 			sc->caps[0].minspeed = sc->caps[0].maxspeed = sc->speed;
1697 			sc->caps[1].minspeed = sc->caps[1].maxspeed = sc->speed;
1698 		}
1699 		else if (ch->speed != 0 && ch->speed != sc->speed) {
1700 			error = -1;
1701 			goto fail;
1702 		}
1703 		if (ch->speed == 0)
1704 			ch->channel->speed = sc->speed;
1705 		/* start or enable channel */
1706 		sc->run[slot]++;
1707 		if (sc->run[slot] == 1) {
1708 			/* first channel */
1709 			ch->offset = 0;
1710 			sc->blk[slot] = ch->blk;
1711 		}
1712 		else {
1713 			ptr = envy24ht_gethwptr(sc, ch->dir);
1714 			ch->offset = ((ptr / ch->blk + 1) * ch->blk %
1715 			    (ch->size / 4)) * 4 / ch->unit;
1716 			if (ch->blk < sc->blk[slot])
1717 				sc->blk[slot] = ch->blk;
1718 		}
1719 		if (ch->dir == PCMDIR_PLAY) {
1720 			ch->emldma(ch);
1721 			envy24ht_setvolume(sc, ch->num);
1722 		}
1723 		envy24ht_updintr(sc, ch->dir);
1724 		if (sc->run[slot] == 1)
1725 			envy24ht_start(sc, ch->dir);
1726 		ch->run = 1;
1727 		break;
1728 	case PCMTRIG_EMLDMAWR:
1729 #if(0)
1730 		device_printf(sc->dev, "envy24htchan_trigger(): emldmawr\n");
1731 #endif
1732 		if (ch->run != 1) {
1733 			error = -1;
1734 			goto fail;
1735 		}
1736 		ch->emldma(ch);
1737 		break;
1738 	case PCMTRIG_EMLDMARD:
1739 #if(0)
1740 		device_printf(sc->dev, "envy24htchan_trigger(): emldmard\n");
1741 #endif
1742 		if (ch->run != 1) {
1743 			error = -1;
1744 			goto fail;
1745 		}
1746 		ch->emldma(ch);
1747 		break;
1748 	case PCMTRIG_ABORT:
1749 		if (ch->run) {
1750 #if(0)
1751 		device_printf(sc->dev, "envy24htchan_trigger(): abort\n");
1752 #endif
1753 		ch->run = 0;
1754 		sc->run[slot]--;
1755 		if (ch->dir == PCMDIR_PLAY)
1756 			envy24ht_mutevolume(sc, ch->num);
1757 		if (sc->run[slot] == 0) {
1758 			envy24ht_stop(sc, ch->dir);
1759 			sc->intr[slot] = 0;
1760 		}
1761 /*		else if (ch->blk == sc->blk[slot]) {
1762 			sc->blk[slot] = ENVY24HT_SAMPLE_NUM / 2;
1763 			for (i = 0; i < ENVY24HT_CHAN_NUM; i++) {
1764 				if (sc->chan[i].dir == ch->dir &&
1765 				    sc->chan[i].run == 1 &&
1766 				    sc->chan[i].blk < sc->blk[slot])
1767 					sc->blk[slot] = sc->chan[i].blk;
1768 			}
1769 			if (ch->blk != sc->blk[slot])
1770 				envy24ht_updintr(sc, ch->dir);
1771 		}*/
1772 		}
1773 		break;
1774 	}
1775 fail:
1776 	snd_mtxunlock(sc->lock);
1777 	return (error);
1778 }
1779 
1780 static u_int32_t
1781 envy24htchan_getptr(kobj_t obj, void *data)
1782 {
1783 	struct sc_chinfo *ch = data;
1784 	struct sc_info *sc = ch->parent;
1785 	u_int32_t ptr, rtn;
1786 
1787 #if(0)
1788 	device_printf(sc->dev, "envy24htchan_getptr()\n");
1789 #endif
1790 	snd_mtxlock(sc->lock);
1791 	ptr = envy24ht_gethwptr(sc, ch->dir);
1792 	rtn = ptr * ch->unit;
1793 	snd_mtxunlock(sc->lock);
1794 
1795 #if(0)
1796 	device_printf(sc->dev, "envy24htchan_getptr(): return %d\n",
1797 	    rtn);
1798 #endif
1799 	return rtn;
1800 }
1801 
1802 static struct pcmchan_caps *
1803 envy24htchan_getcaps(kobj_t obj, void *data)
1804 {
1805 	struct sc_chinfo *ch = data;
1806 	struct sc_info *sc = ch->parent;
1807 	struct pcmchan_caps *rtn;
1808 
1809 #if(0)
1810 	device_printf(sc->dev, "envy24htchan_getcaps()\n");
1811 #endif
1812 	snd_mtxlock(sc->lock);
1813 	if (ch->dir == PCMDIR_PLAY) {
1814 		if (sc->run[0] == 0)
1815 			rtn = &envy24ht_playcaps;
1816 		else
1817 			rtn = &sc->caps[0];
1818 	}
1819 	else {
1820 		if (sc->run[1] == 0)
1821 			rtn = &envy24ht_reccaps;
1822 		else
1823 			rtn = &sc->caps[1];
1824 	}
1825 	snd_mtxunlock(sc->lock);
1826 
1827 	return rtn;
1828 }
1829 
1830 static kobj_method_t envy24htchan_methods[] = {
1831 	KOBJMETHOD(channel_init,		envy24htchan_init),
1832 	KOBJMETHOD(channel_free,		envy24htchan_free),
1833 	KOBJMETHOD(channel_setformat,		envy24htchan_setformat),
1834 	KOBJMETHOD(channel_setspeed,		envy24htchan_setspeed),
1835 	KOBJMETHOD(channel_setblocksize,	envy24htchan_setblocksize),
1836 	KOBJMETHOD(channel_trigger,		envy24htchan_trigger),
1837 	KOBJMETHOD(channel_getptr,		envy24htchan_getptr),
1838 	KOBJMETHOD(channel_getcaps,		envy24htchan_getcaps),
1839 	KOBJMETHOD_END
1840 };
1841 CHANNEL_DECLARE(envy24htchan);
1842 
1843 /* -------------------------------------------------------------------- */
1844 
1845 /* mixer interface */
1846 
1847 static int
1848 envy24htmixer_init(struct snd_mixer *m)
1849 {
1850 	struct sc_info *sc = mix_getdevinfo(m);
1851 
1852 #if(0)
1853 	device_printf(sc->dev, "envy24htmixer_init()\n");
1854 #endif
1855 	if (sc == NULL)
1856 		return -1;
1857 
1858 	/* set volume control rate */
1859 	snd_mtxlock(sc->lock);
1860 #if 0
1861 	envy24ht_wrmt(sc, ENVY24HT_MT_VOLRATE, 0x30, 1); /* 0x30 is default value */
1862 #endif
1863 
1864 	pcm_setflags(sc->dev, pcm_getflags(sc->dev) | SD_F_SOFTPCMVOL);
1865 
1866 	mix_setdevs(m, ENVY24HT_MIX_MASK);
1867 	mix_setrecdevs(m, ENVY24HT_MIX_REC_MASK);
1868 
1869 	snd_mtxunlock(sc->lock);
1870 
1871 	return 0;
1872 }
1873 
1874 static int
1875 envy24htmixer_reinit(struct snd_mixer *m)
1876 {
1877 	struct sc_info *sc = mix_getdevinfo(m);
1878 
1879 	if (sc == NULL)
1880 		return -1;
1881 #if(0)
1882 	device_printf(sc->dev, "envy24htmixer_reinit()\n");
1883 #endif
1884 
1885 	return 0;
1886 }
1887 
1888 static int
1889 envy24htmixer_uninit(struct snd_mixer *m)
1890 {
1891 	struct sc_info *sc = mix_getdevinfo(m);
1892 
1893 	if (sc == NULL)
1894 		return -1;
1895 #if(0)
1896 	device_printf(sc->dev, "envy24htmixer_uninit()\n");
1897 #endif
1898 
1899 	return 0;
1900 }
1901 
1902 static int
1903 envy24htmixer_set(struct snd_mixer *m, unsigned dev, unsigned left, unsigned right)
1904 {
1905 	struct sc_info *sc = mix_getdevinfo(m);
1906 	int ch = envy24ht_mixmap[dev];
1907 	int hwch;
1908 	int i;
1909 
1910 	if (sc == NULL)
1911 		return -1;
1912 	if (dev == 0 && sc->cfg->codec->setvolume == NULL)
1913 		return -1;
1914 	if (dev != 0 && ch == -1)
1915 		return -1;
1916 	hwch = envy24ht_chanmap[ch];
1917 #if(0)
1918 	device_printf(sc->dev, "envy24htmixer_set(m, %d, %d, %d)\n",
1919 	    dev, left, right);
1920 #endif
1921 
1922 	snd_mtxlock(sc->lock);
1923 	if (dev == 0) {
1924 		for (i = 0; i < sc->dacn; i++) {
1925 			sc->cfg->codec->setvolume(sc->dac[i], PCMDIR_PLAY, left, right);
1926 		}
1927 	}
1928 	else {
1929 		/* set volume value for hardware */
1930 		if ((sc->left[hwch] = 100 - left) > ENVY24HT_VOL_MIN)
1931 			sc->left[hwch] = ENVY24HT_VOL_MUTE;
1932 		if ((sc->right[hwch] = 100 - right) > ENVY24HT_VOL_MIN)
1933 			sc->right[hwch] = ENVY24HT_VOL_MUTE;
1934 
1935 		/* set volume for record channel and running play channel */
1936 		if (hwch > ENVY24HT_CHAN_PLAY_SPDIF || sc->chan[ch].run)
1937 			envy24ht_setvolume(sc, hwch);
1938 	}
1939 	snd_mtxunlock(sc->lock);
1940 
1941 	return right << 8 | left;
1942 }
1943 
1944 static u_int32_t
1945 envy24htmixer_setrecsrc(struct snd_mixer *m, u_int32_t src)
1946 {
1947 	struct sc_info *sc = mix_getdevinfo(m);
1948 	int ch = envy24ht_mixmap[src];
1949 #if(0)
1950 	device_printf(sc->dev, "envy24htmixer_setrecsrc(m, %d)\n", src);
1951 #endif
1952 
1953 	if (ch > ENVY24HT_CHAN_PLAY_SPDIF)
1954 		sc->src = ch;
1955 	return src;
1956 }
1957 
1958 static kobj_method_t envy24htmixer_methods[] = {
1959 	KOBJMETHOD(mixer_init,		envy24htmixer_init),
1960 	KOBJMETHOD(mixer_reinit,	envy24htmixer_reinit),
1961 	KOBJMETHOD(mixer_uninit,	envy24htmixer_uninit),
1962 	KOBJMETHOD(mixer_set,		envy24htmixer_set),
1963 	KOBJMETHOD(mixer_setrecsrc,	envy24htmixer_setrecsrc),
1964 	KOBJMETHOD_END
1965 };
1966 MIXER_DECLARE(envy24htmixer);
1967 
1968 /* -------------------------------------------------------------------- */
1969 
1970 /* The interrupt handler */
1971 static void
1972 envy24ht_intr(void *p)
1973 {
1974 	struct sc_info *sc = (struct sc_info *)p;
1975 	struct sc_chinfo *ch;
1976 	u_int32_t ptr, dsize, feed;
1977 	int i;
1978 
1979 #if(0)
1980 	device_printf(sc->dev, "envy24ht_intr()\n");
1981 #endif
1982 	snd_mtxlock(sc->lock);
1983 	if (envy24ht_checkintr(sc, PCMDIR_PLAY)) {
1984 #if(0)
1985 		device_printf(sc->dev, "envy24ht_intr(): play\n");
1986 #endif
1987 		dsize = sc->psize / 4;
1988 		ptr = dsize - envy24ht_rdmt(sc, ENVY24HT_MT_PCNT, 2) - 1;
1989 #if(0)
1990 		device_printf(sc->dev, "envy24ht_intr(): ptr = %d-->", ptr);
1991 #endif
1992 		ptr -= ptr % sc->blk[0];
1993 		feed = (ptr + dsize - sc->intr[0]) % dsize;
1994 #if(0)
1995 		printf("%d intr = %d feed = %d\n", ptr, sc->intr[0], feed);
1996 #endif
1997 		for (i = ENVY24HT_CHAN_PLAY_DAC1; i <= ENVY24HT_CHAN_PLAY_SPDIF; i++) {
1998 			ch = &sc->chan[i];
1999 #if(0)
2000 			if (ch->run)
2001 				device_printf(sc->dev, "envy24ht_intr(): chan[%d].blk = %d\n", i, ch->blk);
2002 #endif
2003 			if (ch->run && ch->blk <= feed) {
2004 				snd_mtxunlock(sc->lock);
2005 				chn_intr(ch->channel);
2006 				snd_mtxlock(sc->lock);
2007 			}
2008 		}
2009 		sc->intr[0] = ptr;
2010 		envy24ht_updintr(sc, PCMDIR_PLAY);
2011 	}
2012 	if (envy24ht_checkintr(sc, PCMDIR_REC)) {
2013 #if(0)
2014 		device_printf(sc->dev, "envy24ht_intr(): rec\n");
2015 #endif
2016 		dsize = sc->rsize / 4;
2017 		ptr = dsize - envy24ht_rdmt(sc, ENVY24HT_MT_RCNT, 2) - 1;
2018 		ptr -= ptr % sc->blk[1];
2019 		feed = (ptr + dsize - sc->intr[1]) % dsize;
2020 		for (i = ENVY24HT_CHAN_REC_ADC1; i <= ENVY24HT_CHAN_REC_SPDIF; i++) {
2021 			ch = &sc->chan[i];
2022 			if (ch->run && ch->blk <= feed) {
2023 				snd_mtxunlock(sc->lock);
2024 				chn_intr(ch->channel);
2025 				snd_mtxlock(sc->lock);
2026 			}
2027 		}
2028 		sc->intr[1] = ptr;
2029 		envy24ht_updintr(sc, PCMDIR_REC);
2030 	}
2031 	snd_mtxunlock(sc->lock);
2032 
2033 	return;
2034 }
2035 
2036 /*
2037  * Probe and attach the card
2038  */
2039 
2040 static int
2041 envy24ht_pci_probe(device_t dev)
2042 {
2043 	u_int16_t sv, sd;
2044 	int i;
2045 
2046 #if(0)
2047 	printf("envy24ht_pci_probe()\n");
2048 #endif
2049 	if (pci_get_device(dev) == PCID_ENVY24HT &&
2050 	    pci_get_vendor(dev) == PCIV_ENVY24) {
2051 		sv = pci_get_subvendor(dev);
2052 		sd = pci_get_subdevice(dev);
2053 		for (i = 0; cfg_table[i].subvendor != 0 || cfg_table[i].subdevice != 0; i++) {
2054 			if (cfg_table[i].subvendor == sv &&
2055 			    cfg_table[i].subdevice == sd) {
2056 				break;
2057 			}
2058 		}
2059 		device_set_desc(dev, cfg_table[i].name);
2060 #if(0)
2061 		printf("envy24ht_pci_probe(): return 0\n");
2062 #endif
2063 		return 0;
2064 	}
2065 	else {
2066 #if(0)
2067 		printf("envy24ht_pci_probe(): return ENXIO\n");
2068 #endif
2069 		return ENXIO;
2070 	}
2071 }
2072 
2073 static void
2074 envy24ht_dmapsetmap(void *arg, bus_dma_segment_t *segs, int nseg, int error)
2075 {
2076 	struct sc_info *sc = arg;
2077 
2078 	sc->paddr = segs->ds_addr;
2079 #if(0)
2080 	device_printf(sc->dev, "envy24ht_dmapsetmap()\n");
2081 	if (bootverbose) {
2082 		printf("envy24ht(play): setmap %lx, %lx; ",
2083 		    (unsigned long)segs->ds_addr,
2084 		    (unsigned long)segs->ds_len);
2085 	}
2086 #endif
2087 	envy24ht_wrmt(sc, ENVY24HT_MT_PADDR, (uint32_t)segs->ds_addr, 4);
2088 	envy24ht_wrmt(sc, ENVY24HT_MT_PCNT, (uint32_t)(segs->ds_len / 4 - 1), 2);
2089 }
2090 
2091 static void
2092 envy24ht_dmarsetmap(void *arg, bus_dma_segment_t *segs, int nseg, int error)
2093 {
2094 	struct sc_info *sc = arg;
2095 
2096 	sc->raddr = segs->ds_addr;
2097 #if(0)
2098 	device_printf(sc->dev, "envy24ht_dmarsetmap()\n");
2099 	if (bootverbose) {
2100 		printf("envy24ht(record): setmap %lx, %lx; ",
2101 		    (unsigned long)segs->ds_addr,
2102 		    (unsigned long)segs->ds_len);
2103 	}
2104 #endif
2105 	envy24ht_wrmt(sc, ENVY24HT_MT_RADDR, (uint32_t)segs->ds_addr, 4);
2106 	envy24ht_wrmt(sc, ENVY24HT_MT_RCNT, (uint32_t)(segs->ds_len / 4 - 1), 2);
2107 }
2108 
2109 static void
2110 envy24ht_dmafree(struct sc_info *sc)
2111 {
2112 #if(0)
2113 	device_printf(sc->dev, "envy24ht_dmafree():");
2114 	printf(" sc->raddr(0x%08x)", (u_int32_t)sc->raddr);
2115 	printf(" sc->paddr(0x%08x)", (u_int32_t)sc->paddr);
2116 	if (sc->rbuf) printf(" sc->rbuf(0x%08x)", (u_int32_t)sc->rbuf);
2117 	else printf(" sc->rbuf(null)");
2118 	if (sc->pbuf) printf(" sc->pbuf(0x%08x)\n", (u_int32_t)sc->pbuf);
2119 	else printf(" sc->pbuf(null)\n");
2120 #endif
2121 #if(0)
2122 	if (sc->raddr)
2123 		bus_dmamap_unload(sc->dmat, sc->rmap);
2124 	if (sc->paddr)
2125 		bus_dmamap_unload(sc->dmat, sc->pmap);
2126 	if (sc->rbuf)
2127 		bus_dmamem_free(sc->dmat, sc->rbuf, sc->rmap);
2128 	if (sc->pbuf)
2129 		bus_dmamem_free(sc->dmat, sc->pbuf, sc->pmap);
2130 #else
2131 	bus_dmamap_unload(sc->dmat, sc->rmap);
2132 	bus_dmamap_unload(sc->dmat, sc->pmap);
2133 	bus_dmamem_free(sc->dmat, sc->rbuf, sc->rmap);
2134 	bus_dmamem_free(sc->dmat, sc->pbuf, sc->pmap);
2135 #endif
2136 
2137 	sc->raddr = sc->paddr = 0;
2138 	sc->pbuf = NULL;
2139 	sc->rbuf = NULL;
2140 
2141 	return;
2142 }
2143 
2144 static int
2145 envy24ht_dmainit(struct sc_info *sc)
2146 {
2147 
2148 #if(0)
2149 	device_printf(sc->dev, "envy24ht_dmainit()\n");
2150 #endif
2151 	/* init values */
2152 	sc->psize = ENVY24HT_PLAY_BUFUNIT * ENVY24HT_SAMPLE_NUM;
2153 	sc->rsize = ENVY24HT_REC_BUFUNIT * ENVY24HT_SAMPLE_NUM;
2154 	sc->pbuf = NULL;
2155 	sc->rbuf = NULL;
2156 	sc->paddr = sc->raddr = 0;
2157 	sc->blk[0] = sc->blk[1] = 0;
2158 
2159 	/* allocate DMA buffer */
2160 #if(0)
2161 	device_printf(sc->dev, "envy24ht_dmainit(): bus_dmamem_alloc(): sc->pbuf\n");
2162 #endif
2163 	if (bus_dmamem_alloc(sc->dmat, (void **)&sc->pbuf, BUS_DMA_NOWAIT, &sc->pmap))
2164 		goto bad;
2165 #if(0)
2166 	device_printf(sc->dev, "envy24ht_dmainit(): bus_dmamem_alloc(): sc->rbuf\n");
2167 #endif
2168 	if (bus_dmamem_alloc(sc->dmat, (void **)&sc->rbuf, BUS_DMA_NOWAIT, &sc->rmap))
2169 		goto bad;
2170 #if(0)
2171 	device_printf(sc->dev, "envy24ht_dmainit(): bus_dmamem_load(): sc->pmap\n");
2172 #endif
2173 	if (bus_dmamap_load(sc->dmat, sc->pmap, sc->pbuf, sc->psize, envy24ht_dmapsetmap, sc, BUS_DMA_NOWAIT))
2174 		goto bad;
2175 #if(0)
2176 	device_printf(sc->dev, "envy24ht_dmainit(): bus_dmamem_load(): sc->rmap\n");
2177 #endif
2178 	if (bus_dmamap_load(sc->dmat, sc->rmap, sc->rbuf, sc->rsize, envy24ht_dmarsetmap, sc, BUS_DMA_NOWAIT))
2179 		goto bad;
2180 	bzero(sc->pbuf, sc->psize);
2181 	bzero(sc->rbuf, sc->rsize);
2182 
2183 	return 0;
2184  bad:
2185 	envy24ht_dmafree(sc);
2186 	return ENOSPC;
2187 }
2188 
2189 static void
2190 envy24ht_putcfg(struct sc_info *sc)
2191 {
2192 	device_printf(sc->dev, "system configuration\n");
2193 	printf("  SubVendorID: 0x%04x, SubDeviceID: 0x%04x\n",
2194 	    sc->cfg->subvendor, sc->cfg->subdevice);
2195 	printf("  XIN2 Clock Source: ");
2196 	switch (sc->cfg->scfg & ENVY24HT_CCSM_SCFG_XIN2) {
2197 	case 0x00:
2198 		printf("24.576MHz(96kHz*256)\n");
2199 		break;
2200 	case 0x40:
2201 		printf("49.152MHz(192kHz*256)\n");
2202 		break;
2203 	case 0x80:
2204 		printf("reserved\n");
2205 		break;
2206 	default:
2207 		printf("illegal system setting\n");
2208 	}
2209 	printf("  MPU-401 UART(s) #: ");
2210 	if (sc->cfg->scfg & ENVY24HT_CCSM_SCFG_MPU)
2211 		printf("1\n");
2212 	else
2213 		printf("not implemented\n");
2214         switch (sc->adcn) {
2215         case 0x01:
2216 	case 0x02:
2217                 printf("  ADC #: ");
2218                 printf("%d\n", sc->adcn);
2219                 break;
2220         case 0x03:
2221                 printf("  ADC #: ");
2222                 printf("%d", 1);
2223                 printf(" and SPDIF receiver connected\n");
2224                 break;
2225         default:
2226                 printf("  no physical inputs\n");
2227         }
2228 	printf("  DAC #: ");
2229 	printf("%d\n", sc->dacn);
2230 	printf("  Multi-track converter type: ");
2231 	if ((sc->cfg->acl & ENVY24HT_CCSM_ACL_MTC) == 0) {
2232 		printf("AC'97(SDATA_OUT:");
2233 		if (sc->cfg->acl & ENVY24HT_CCSM_ACL_OMODE)
2234 			printf("packed");
2235 		else
2236 			printf("split");
2237 		printf(")\n");
2238 	}
2239 	else {
2240 		printf("I2S(");
2241 		if (sc->cfg->i2s & ENVY24HT_CCSM_I2S_VOL)
2242 			printf("with volume, ");
2243                 if (sc->cfg->i2s & ENVY24HT_CCSM_I2S_192KHZ)
2244                         printf("192KHz support, ");
2245                 else
2246                 if (sc->cfg->i2s & ENVY24HT_CCSM_I2S_96KHZ)
2247                         printf("192KHz support, ");
2248                 else
2249                         printf("48KHz support, ");
2250 		switch (sc->cfg->i2s & ENVY24HT_CCSM_I2S_RES) {
2251 		case ENVY24HT_CCSM_I2S_16BIT:
2252 			printf("16bit resolution, ");
2253 			break;
2254 		case ENVY24HT_CCSM_I2S_18BIT:
2255 			printf("18bit resolution, ");
2256 			break;
2257 		case ENVY24HT_CCSM_I2S_20BIT:
2258 			printf("20bit resolution, ");
2259 			break;
2260 		case ENVY24HT_CCSM_I2S_24BIT:
2261 			printf("24bit resolution, ");
2262 			break;
2263 		}
2264 		printf("ID#0x%x)\n", sc->cfg->i2s & ENVY24HT_CCSM_I2S_ID);
2265 	}
2266 	printf("  S/PDIF(IN/OUT): ");
2267 	if (sc->cfg->spdif & ENVY24HT_CCSM_SPDIF_IN)
2268 		printf("1/");
2269 	else
2270 		printf("0/");
2271 	if (sc->cfg->spdif & ENVY24HT_CCSM_SPDIF_OUT)
2272 		printf("1 ");
2273 	else
2274 		printf("0 ");
2275 	if (sc->cfg->spdif & (ENVY24HT_CCSM_SPDIF_IN | ENVY24HT_CCSM_SPDIF_OUT))
2276 		printf("ID# 0x%02x\n", (sc->cfg->spdif & ENVY24HT_CCSM_SPDIF_ID) >> 2);
2277 	printf("  GPIO(mask/dir/state): 0x%02x/0x%02x/0x%02x\n",
2278 	    sc->cfg->gpiomask, sc->cfg->gpiodir, sc->cfg->gpiostate);
2279 }
2280 
2281 static int
2282 envy24ht_init(struct sc_info *sc)
2283 {
2284 	u_int32_t data;
2285 #if(0)
2286 	int rtn;
2287 #endif
2288 	int i;
2289 	u_int32_t sv, sd;
2290 
2291 #if(0)
2292 	device_printf(sc->dev, "envy24ht_init()\n");
2293 #endif
2294 
2295 	/* reset chip */
2296 #if 0
2297 	envy24ht_wrcs(sc, ENVY24HT_CCS_CTL, ENVY24HT_CCS_CTL_RESET, 1);
2298 	DELAY(200);
2299 	envy24ht_wrcs(sc, ENVY24HT_CCS_CTL, ENVY24HT_CCS_CTL_NATIVE, 1);
2300 	DELAY(200);
2301 
2302 	/* legacy hardware disable */
2303 	data = pci_read_config(sc->dev, PCIR_LAC, 2);
2304 	data |= PCIM_LAC_DISABLE;
2305 	pci_write_config(sc->dev, PCIR_LAC, data, 2);
2306 #endif
2307 
2308 	/* check system configuration */
2309 	sc->cfg = NULL;
2310 	for (i = 0; cfg_table[i].subvendor != 0 || cfg_table[i].subdevice != 0; i++) {
2311 		/* 1st: search configuration from table */
2312 		sv = pci_get_subvendor(sc->dev);
2313 		sd = pci_get_subdevice(sc->dev);
2314 		if (sv == cfg_table[i].subvendor && sd == cfg_table[i].subdevice) {
2315 #if(0)
2316 			device_printf(sc->dev, "Set configuration from table\n");
2317 #endif
2318 			sc->cfg = &cfg_table[i];
2319 			break;
2320 		}
2321 	}
2322 	if (sc->cfg == NULL) {
2323 		/* 2nd: read configuration from table */
2324 		sc->cfg = envy24ht_rom2cfg(sc);
2325 	}
2326 	sc->adcn = ((sc->cfg->scfg & ENVY24HT_CCSM_SCFG_ADC) >> 2) + 1; /* need to be fixed */
2327 	sc->dacn = (sc->cfg->scfg & ENVY24HT_CCSM_SCFG_DAC) + 1;
2328 
2329 	if (1 /* bootverbose */) {
2330 		envy24ht_putcfg(sc);
2331 	}
2332 
2333 	/* set system configuration */
2334 	envy24ht_wrcs(sc, ENVY24HT_CCS_SCFG, sc->cfg->scfg, 1);
2335 	envy24ht_wrcs(sc, ENVY24HT_CCS_ACL, sc->cfg->acl, 1);
2336 	envy24ht_wrcs(sc, ENVY24HT_CCS_I2S, sc->cfg->i2s, 1);
2337 	envy24ht_wrcs(sc, ENVY24HT_CCS_SPDIF, sc->cfg->spdif, 1);
2338 	envy24ht_gpiosetmask(sc, sc->cfg->gpiomask);
2339 	envy24ht_gpiosetdir(sc, sc->cfg->gpiodir);
2340 	envy24ht_gpiowr(sc, sc->cfg->gpiostate);
2341 
2342 	if ((sc->cfg->subvendor == 0x3031) && (sc->cfg->subdevice == 0x4553)) {
2343 		envy24ht_wri2c(sc, 0x22, 0x00, 0x07);
2344 		envy24ht_wri2c(sc, 0x22, 0x04, 0x5f | 0x80);
2345 		envy24ht_wri2c(sc, 0x22, 0x05, 0x5f | 0x80);
2346 	}
2347 
2348 	for (i = 0; i < sc->adcn; i++) {
2349 		sc->adc[i] = sc->cfg->codec->create(sc->dev, sc, PCMDIR_REC, i);
2350 		sc->cfg->codec->init(sc->adc[i]);
2351 	}
2352 	for (i = 0; i < sc->dacn; i++) {
2353 		sc->dac[i] = sc->cfg->codec->create(sc->dev, sc, PCMDIR_PLAY, i);
2354 		sc->cfg->codec->init(sc->dac[i]);
2355 	}
2356 
2357 	/* initialize DMA buffer */
2358 #if(0)
2359 	device_printf(sc->dev, "envy24ht_init(): initialize DMA buffer\n");
2360 #endif
2361 	if (envy24ht_dmainit(sc))
2362 		return ENOSPC;
2363 
2364 	/* initialize status */
2365 	sc->run[0] = sc->run[1] = 0;
2366 	sc->intr[0] = sc->intr[1] = 0;
2367 	sc->speed = 0;
2368 	sc->caps[0].fmtlist = envy24ht_playfmt;
2369 	sc->caps[1].fmtlist = envy24ht_recfmt;
2370 
2371 	/* set channel router */
2372 #if 0
2373 	envy24ht_route(sc, ENVY24HT_ROUTE_DAC_1, ENVY24HT_ROUTE_CLASS_MIX, 0, 0);
2374 	envy24ht_route(sc, ENVY24HT_ROUTE_DAC_SPDIF, ENVY24HT_ROUTE_CLASS_DMA, 0, 0);
2375 	envy24ht_route(sc, ENVY24HT_ROUTE_DAC_SPDIF, ENVY24HT_ROUTE_CLASS_MIX, 0, 0);
2376 #endif
2377 
2378 	/* set macro interrupt mask */
2379 	data = envy24ht_rdcs(sc, ENVY24HT_CCS_IMASK, 1);
2380 	envy24ht_wrcs(sc, ENVY24HT_CCS_IMASK, data & ~ENVY24HT_CCS_IMASK_PMT, 1);
2381 	data = envy24ht_rdcs(sc, ENVY24HT_CCS_IMASK, 1);
2382 #if(0)
2383 	device_printf(sc->dev, "envy24ht_init(): CCS_IMASK-->0x%02x\n", data);
2384 #endif
2385 
2386 	return 0;
2387 }
2388 
2389 static int
2390 envy24ht_alloc_resource(struct sc_info *sc)
2391 {
2392 	/* allocate I/O port resource */
2393 	sc->csid = PCIR_CCS;
2394 	sc->cs = bus_alloc_resource_any(sc->dev, SYS_RES_IOPORT,
2395 	    &sc->csid, RF_ACTIVE);
2396 	sc->mtid = ENVY24HT_PCIR_MT;
2397 	sc->mt = bus_alloc_resource_any(sc->dev, SYS_RES_IOPORT,
2398 	    &sc->mtid, RF_ACTIVE);
2399 	if (!sc->cs || !sc->mt) {
2400 		device_printf(sc->dev, "unable to map IO port space\n");
2401 		return ENXIO;
2402 	}
2403 	sc->cst = rman_get_bustag(sc->cs);
2404 	sc->csh = rman_get_bushandle(sc->cs);
2405 	sc->mtt = rman_get_bustag(sc->mt);
2406 	sc->mth = rman_get_bushandle(sc->mt);
2407 #if(0)
2408 	device_printf(sc->dev,
2409 	    "IO port register values\nCCS: 0x%lx\nMT: 0x%lx\n",
2410 	    pci_read_config(sc->dev, PCIR_CCS, 4),
2411 	    pci_read_config(sc->dev, PCIR_MT, 4));
2412 #endif
2413 
2414 	/* allocate interrupt resource */
2415 	sc->irqid = 0;
2416 	sc->irq = bus_alloc_resource_any(sc->dev, SYS_RES_IRQ, &sc->irqid,
2417 				 RF_ACTIVE | RF_SHAREABLE);
2418 	if (!sc->irq ||
2419 	    snd_setup_intr(sc->dev, sc->irq, INTR_MPSAFE, envy24ht_intr, sc, &sc->ih)) {
2420 		device_printf(sc->dev, "unable to map interrupt\n");
2421 		return ENXIO;
2422 	}
2423 
2424 	/* allocate DMA resource */
2425 	if (bus_dma_tag_create(/*parent*/bus_get_dma_tag(sc->dev),
2426 	    /*alignment*/4,
2427 	    /*boundary*/0,
2428 	    /*lowaddr*/BUS_SPACE_MAXADDR_32BIT,
2429 	    /*highaddr*/BUS_SPACE_MAXADDR,
2430 	    /*filter*/NULL, /*filterarg*/NULL,
2431 	    /*maxsize*/BUS_SPACE_MAXSIZE_ENVY24,
2432 	    /*nsegments*/1, /*maxsegsz*/0x3ffff,
2433 	    /*flags*/0, /*lockfunc*/NULL,
2434 	    /*lockarg*/NULL, &sc->dmat) != 0) {
2435 		device_printf(sc->dev, "unable to create dma tag\n");
2436 		return ENXIO;
2437 	}
2438 
2439 	return 0;
2440 }
2441 
2442 static int
2443 envy24ht_pci_attach(device_t dev)
2444 {
2445 	struct sc_info 		*sc;
2446 	char 			status[SND_STATUSLEN];
2447 	int			err = 0;
2448 	int			i;
2449 
2450 #if(0)
2451 	device_printf(dev, "envy24ht_pci_attach()\n");
2452 #endif
2453 	/* get sc_info data area */
2454 	if ((sc = malloc(sizeof(*sc), M_ENVY24HT, M_NOWAIT)) == NULL) {
2455 		device_printf(dev, "cannot allocate softc\n");
2456 		return ENXIO;
2457 	}
2458 
2459 	bzero(sc, sizeof(*sc));
2460 	sc->lock = snd_mtxcreate(device_get_nameunit(dev),
2461 	    "snd_envy24ht softc");
2462 	sc->dev = dev;
2463 
2464 	/* initialize PCI interface */
2465 	pci_enable_busmaster(dev);
2466 
2467 	/* allocate resources */
2468 	err = envy24ht_alloc_resource(sc);
2469 	if (err) {
2470 		device_printf(dev, "unable to allocate system resources\n");
2471 		goto bad;
2472 	}
2473 
2474 	/* initialize card */
2475 	err = envy24ht_init(sc);
2476 	if (err) {
2477 		device_printf(dev, "unable to initialize the card\n");
2478 		goto bad;
2479 	}
2480 
2481 	/* set multi track mixer */
2482 	mixer_init(dev, &envy24htmixer_class, sc);
2483 
2484 	/* set channel information */
2485 	/* err = pcm_register(dev, sc, 5, 2 + sc->adcn); */
2486 	err = pcm_register(dev, sc, 1, 2 + sc->adcn);
2487 	if (err)
2488 		goto bad;
2489 	sc->chnum = 0;
2490 	/* for (i = 0; i < 5; i++) { */
2491 		pcm_addchan(dev, PCMDIR_PLAY, &envy24htchan_class, sc);
2492 		sc->chnum++;
2493 	/* } */
2494 	for (i = 0; i < 2 + sc->adcn; i++) {
2495 		pcm_addchan(dev, PCMDIR_REC, &envy24htchan_class, sc);
2496 		sc->chnum++;
2497 	}
2498 
2499 	/* set status iformation */
2500 	snprintf(status, SND_STATUSLEN,
2501 	    "at io 0x%jx:%jd,0x%jx:%jd irq %jd",
2502 	    rman_get_start(sc->cs),
2503 	    rman_get_end(sc->cs) - rman_get_start(sc->cs) + 1,
2504 	    rman_get_start(sc->mt),
2505 	    rman_get_end(sc->mt) - rman_get_start(sc->mt) + 1,
2506 	    rman_get_start(sc->irq));
2507 	pcm_setstatus(dev, status);
2508 
2509 	return 0;
2510 
2511 bad:
2512 	if (sc->ih)
2513 		bus_teardown_intr(dev, sc->irq, sc->ih);
2514 	if (sc->irq)
2515 		bus_release_resource(dev, SYS_RES_IRQ, sc->irqid, sc->irq);
2516 	envy24ht_dmafree(sc);
2517 	if (sc->dmat)
2518 		bus_dma_tag_destroy(sc->dmat);
2519         if (sc->cfg->codec->destroy != NULL) {
2520                 for (i = 0; i < sc->adcn; i++)
2521                         sc->cfg->codec->destroy(sc->adc[i]);
2522                 for (i = 0; i < sc->dacn; i++)
2523                         sc->cfg->codec->destroy(sc->dac[i]);
2524         }
2525 	envy24ht_cfgfree(sc->cfg);
2526 	if (sc->cs)
2527 		bus_release_resource(dev, SYS_RES_IOPORT, sc->csid, sc->cs);
2528 	if (sc->mt)
2529 		bus_release_resource(dev, SYS_RES_IOPORT, sc->mtid, sc->mt);
2530 	if (sc->lock)
2531 		snd_mtxfree(sc->lock);
2532 	free(sc, M_ENVY24HT);
2533 	return err;
2534 }
2535 
2536 static int
2537 envy24ht_pci_detach(device_t dev)
2538 {
2539 	struct sc_info *sc;
2540 	int r;
2541 	int i;
2542 
2543 #if(0)
2544 	device_printf(dev, "envy24ht_pci_detach()\n");
2545 #endif
2546 	sc = pcm_getdevinfo(dev);
2547 	if (sc == NULL)
2548 		return 0;
2549 	r = pcm_unregister(dev);
2550 	if (r)
2551 		return r;
2552 
2553 	envy24ht_dmafree(sc);
2554 	if (sc->cfg->codec->destroy != NULL) {
2555 		for (i = 0; i < sc->adcn; i++)
2556 			sc->cfg->codec->destroy(sc->adc[i]);
2557 		for (i = 0; i < sc->dacn; i++)
2558 			sc->cfg->codec->destroy(sc->dac[i]);
2559 	}
2560 	envy24ht_cfgfree(sc->cfg);
2561 	bus_dma_tag_destroy(sc->dmat);
2562 	bus_teardown_intr(dev, sc->irq, sc->ih);
2563 	bus_release_resource(dev, SYS_RES_IRQ, sc->irqid, sc->irq);
2564 	bus_release_resource(dev, SYS_RES_IOPORT, sc->csid, sc->cs);
2565 	bus_release_resource(dev, SYS_RES_IOPORT, sc->mtid, sc->mt);
2566 	snd_mtxfree(sc->lock);
2567 	free(sc, M_ENVY24HT);
2568 	return 0;
2569 }
2570 
2571 static device_method_t envy24ht_methods[] = {
2572 	/* Device interface */
2573 	DEVMETHOD(device_probe,		envy24ht_pci_probe),
2574 	DEVMETHOD(device_attach,	envy24ht_pci_attach),
2575 	DEVMETHOD(device_detach,	envy24ht_pci_detach),
2576 	{ 0, 0 }
2577 };
2578 
2579 static driver_t envy24ht_driver = {
2580 	"pcm",
2581 	envy24ht_methods,
2582 	PCM_SOFTC_SIZE,
2583 };
2584 
2585 DRIVER_MODULE(snd_envy24ht, pci, envy24ht_driver, pcm_devclass, 0, 0);
2586 MODULE_DEPEND(snd_envy24ht, sound, SOUND_MINVER, SOUND_PREFVER, SOUND_MAXVER);
2587 MODULE_DEPEND(snd_envy24ht, snd_spicds, 1, 1, 1);
2588 MODULE_VERSION(snd_envy24ht, 1);
2589