tea575x-tuner.c 14.5 KB
Newer Older
L
Linus Torvalds 已提交
1 2 3
/*
 *   ALSA driver for TEA5757/5759 Philips AM/FM radio tuner chips
 *
4
 *	Copyright (c) 2004 Jaroslav Kysela <perex@perex.cz>
L
Linus Torvalds 已提交
5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20
 *
 *
 *   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
 *
21
 */
L
Linus Torvalds 已提交
22 23 24

#include <asm/io.h>
#include <linux/delay.h>
25
#include <linux/module.h>
L
Linus Torvalds 已提交
26
#include <linux/init.h>
27
#include <linux/slab.h>
28 29
#include <linux/sched.h>
#include <media/v4l2-device.h>
30
#include <media/v4l2-dev.h>
31
#include <media/v4l2-fh.h>
32
#include <media/v4l2-ioctl.h>
33
#include <media/v4l2-event.h>
34
#include <media/tea575x.h>
L
Linus Torvalds 已提交
35

36
MODULE_AUTHOR("Jaroslav Kysela <perex@perex.cz>");
L
Linus Torvalds 已提交
37 38 39 40 41 42 43 44 45 46 47 48 49
MODULE_DESCRIPTION("Routines for control of TEA5757/5759 Philips AM/FM radio tuner chips");
MODULE_LICENSE("GPL");

/*
 * definitions
 */

#define TEA575X_BIT_SEARCH	(1<<24)		/* 1 = search action, 0 = tuned */
#define TEA575X_BIT_UPDOWN	(1<<23)		/* 0 = search down, 1 = search up */
#define TEA575X_BIT_MONO	(1<<22)		/* 0 = stereo, 1 = mono */
#define TEA575X_BIT_BAND_MASK	(3<<20)
#define TEA575X_BIT_BAND_FM	(0<<20)
#define TEA575X_BIT_BAND_MW	(1<<20)
50 51
#define TEA575X_BIT_BAND_LW	(2<<20)
#define TEA575X_BIT_BAND_SW	(3<<20)
L
Linus Torvalds 已提交
52 53 54 55 56 57 58 59 60 61
#define TEA575X_BIT_PORT_0	(1<<19)		/* user bit */
#define TEA575X_BIT_PORT_1	(1<<18)		/* user bit */
#define TEA575X_BIT_SEARCH_MASK	(3<<16)		/* search level */
#define TEA575X_BIT_SEARCH_5_28	     (0<<16)	/* FM >5uV, AM >28uV */
#define TEA575X_BIT_SEARCH_10_40     (1<<16)	/* FM >10uV, AM > 40uV */
#define TEA575X_BIT_SEARCH_30_63     (2<<16)	/* FM >30uV, AM > 63uV */
#define TEA575X_BIT_SEARCH_150_1000  (3<<16)	/* FM > 150uV, AM > 1000uV */
#define TEA575X_BIT_DUMMY	(1<<15)		/* buffer */
#define TEA575X_BIT_FREQ_MASK	0x7fff

62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92
enum { BAND_FM, BAND_FM_JAPAN, BAND_AM };

static const struct v4l2_frequency_band bands[] = {
	{
		.type = V4L2_TUNER_RADIO,
		.index = 0,
		.capability = V4L2_TUNER_CAP_LOW | V4L2_TUNER_CAP_STEREO |
			      V4L2_TUNER_CAP_FREQ_BANDS,
		.rangelow   =  87500 * 16,
		.rangehigh  = 108000 * 16,
		.modulation = V4L2_BAND_MODULATION_FM,
	},
	{
		.type = V4L2_TUNER_RADIO,
		.index = 0,
		.capability = V4L2_TUNER_CAP_LOW | V4L2_TUNER_CAP_STEREO |
			      V4L2_TUNER_CAP_FREQ_BANDS,
		.rangelow   = 76000 * 16,
		.rangehigh  = 91000 * 16,
		.modulation = V4L2_BAND_MODULATION_FM,
	},
	{
		.type = V4L2_TUNER_RADIO,
		.index = 1,
		.capability = V4L2_TUNER_CAP_LOW | V4L2_TUNER_CAP_FREQ_BANDS,
		.rangelow   =  530 * 16,
		.rangehigh  = 1710 * 16,
		.modulation = V4L2_BAND_MODULATION_AM,
	},
};

L
Linus Torvalds 已提交
93 94 95 96
/*
 * lowlevel part
 */

97 98 99 100 101
static void snd_tea575x_write(struct snd_tea575x *tea, unsigned int val)
{
	u16 l;
	u8 data;

102 103 104
	if (tea->ops->write_val)
		return tea->ops->write_val(tea, val);

105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122
	tea->ops->set_direction(tea, 1);
	udelay(16);

	for (l = 25; l > 0; l--) {
		data = (val >> 24) & TEA575X_DATA;
		val <<= 1;			/* shift data */
		tea->ops->set_pins(tea, data | TEA575X_WREN);
		udelay(2);
		tea->ops->set_pins(tea, data | TEA575X_WREN | TEA575X_CLK);
		udelay(2);
		tea->ops->set_pins(tea, data | TEA575X_WREN);
		udelay(2);
	}

	if (!tea->mute)
		tea->ops->set_pins(tea, 0);
}

H
Hans Verkuil 已提交
123
static u32 snd_tea575x_read(struct snd_tea575x *tea)
124 125 126 127
{
	u16 l, rdata;
	u32 data = 0;

128 129 130
	if (tea->ops->read_val)
		return tea->ops->read_val(tea);

131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156
	tea->ops->set_direction(tea, 0);
	tea->ops->set_pins(tea, 0);
	udelay(16);

	for (l = 24; l--;) {
		tea->ops->set_pins(tea, TEA575X_CLK);
		udelay(2);
		if (!l)
			tea->tuned = tea->ops->get_pins(tea) & TEA575X_MOST ? 0 : 1;
		tea->ops->set_pins(tea, 0);
		udelay(2);
		data <<= 1;			/* shift data */
		rdata = tea->ops->get_pins(tea);
		if (!l)
			tea->stereo = (rdata & TEA575X_MOST) ?  0 : 1;
		if (rdata & TEA575X_DATA)
			data++;
		udelay(2);
	}

	if (tea->mute)
		tea->ops->set_pins(tea, TEA575X_WREN);

	return data;
}

157
static u32 snd_tea575x_val_to_freq(struct snd_tea575x *tea, u32 val)
H
Hans Verkuil 已提交
158
{
159
	u32 freq = val & TEA575X_BIT_FREQ_MASK;
H
Hans Verkuil 已提交
160 161 162 163

	if (freq == 0)
		return freq;

164 165 166 167 168 169
	switch (tea->band) {
	case BAND_FM:
		/* freq *= 12.5 */
		freq *= 125;
		freq /= 10;
		/* crystal fixup */
H
Hans Verkuil 已提交
170
		freq -= TEA575X_FMIF;
171 172 173 174 175 176 177 178 179 180 181 182 183
		break;
	case BAND_FM_JAPAN:
		/* freq *= 12.5 */
		freq *= 125;
		freq /= 10;
		/* crystal fixup */
		freq += TEA575X_FMIF;
		break;
	case BAND_AM:
		/* crystal fixup */
		freq -= TEA575X_AMIF;
		break;
	}
H
Hans Verkuil 已提交
184

185 186
	return clamp(freq * 16, bands[tea->band].rangelow,
				bands[tea->band].rangehigh); /* from kHz */
H
Hans Verkuil 已提交
187 188
}

189 190 191 192 193
static u32 snd_tea575x_get_freq(struct snd_tea575x *tea)
{
	return snd_tea575x_val_to_freq(tea, snd_tea575x_read(tea));
}

194
void snd_tea575x_set_freq(struct snd_tea575x *tea)
L
Linus Torvalds 已提交
195
{
196 197
	u32 freq = tea->freq / 16;	/* to kHz */
	u32 band = 0;
L
Linus Torvalds 已提交
198

199 200 201 202
	switch (tea->band) {
	case BAND_FM:
		band = TEA575X_BIT_BAND_FM;
		/* crystal fixup */
203
		freq += TEA575X_FMIF;
204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221
		/* freq /= 12.5 */
		freq *= 10;
		freq /= 125;
		break;
	case BAND_FM_JAPAN:
		band = TEA575X_BIT_BAND_FM;
		/* crystal fixup */
		freq -= TEA575X_FMIF;
		/* freq /= 12.5 */
		freq *= 10;
		freq /= 125;
		break;
	case BAND_AM:
		band = TEA575X_BIT_BAND_MW;
		/* crystal fixup */
		freq += TEA575X_AMIF;
		break;
	}
L
Linus Torvalds 已提交
222

223 224
	tea->val &= ~(TEA575X_BIT_FREQ_MASK | TEA575X_BIT_BAND_MASK);
	tea->val |= band;
L
Linus Torvalds 已提交
225
	tea->val |= freq & TEA575X_BIT_FREQ_MASK;
226
	snd_tea575x_write(tea, tea->val);
227
	tea->freq = snd_tea575x_val_to_freq(tea, tea->val);
L
Linus Torvalds 已提交
228 229 230 231 232 233
}

/*
 * Linux Video interface
 */

234 235
static int vidioc_querycap(struct file *file, void  *priv,
					struct v4l2_capability *v)
L
Linus Torvalds 已提交
236
{
237
	struct snd_tea575x *tea = video_drvdata(file);
238

239
	strlcpy(v->driver, tea->v4l2_dev->name, sizeof(v->driver));
240 241 242
	strlcpy(v->card, tea->card, sizeof(v->card));
	strlcat(v->card, tea->tea5759 ? " TEA5759" : " TEA5757", sizeof(v->card));
	strlcpy(v->bus_info, tea->bus_info, sizeof(v->bus_info));
243 244 245 246
	v->device_caps = V4L2_CAP_TUNER | V4L2_CAP_RADIO;
	if (!tea->cannot_read_data)
		v->device_caps |= V4L2_CAP_HW_FREQ_SEEK;
	v->capabilities = v->device_caps | V4L2_CAP_DEVICE_CAPS;
247 248 249
	return 0;
}

250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282
static int vidioc_enum_freq_bands(struct file *file, void *priv,
					 struct v4l2_frequency_band *band)
{
	struct snd_tea575x *tea = video_drvdata(file);
	int index;

	if (band->tuner != 0)
		return -EINVAL;

	switch (band->index) {
	case 0:
		if (tea->tea5759)
			index = BAND_FM_JAPAN;
		else
			index = BAND_FM;
		break;
	case 1:
		if (tea->has_am) {
			index = BAND_AM;
			break;
		}
		/* Fall through */
	default:
		return -EINVAL;
	}

	*band = bands[index];
	if (!tea->cannot_read_data)
		band->capability |= V4L2_TUNER_CAP_HWSEEK_BOUNDED;

	return 0;
}

283 284 285
static int vidioc_g_tuner(struct file *file, void *priv,
					struct v4l2_tuner *v)
{
286
	struct snd_tea575x *tea = video_drvdata(file);
287
	struct v4l2_frequency_band band_fm = { 0, };
288

289 290 291
	if (v->index > 0)
		return -EINVAL;

292
	snd_tea575x_read(tea);
293
	vidioc_enum_freq_bands(file, priv, &band_fm);
294

295 296
	memset(v, 0, sizeof(*v));
	strlcpy(v->name, tea->has_am ? "FM/AM" : "FM", sizeof(v->name));
297
	v->type = V4L2_TUNER_RADIO;
298 299 300
	v->capability = band_fm.capability;
	v->rangelow = tea->has_am ? bands[BAND_AM].rangelow : band_fm.rangelow;
	v->rangehigh = band_fm.rangehigh;
301 302 303
	v->rxsubchans = tea->stereo ? V4L2_TUNER_SUB_STEREO : V4L2_TUNER_SUB_MONO;
	v->audmode = (tea->val & TEA575X_BIT_MONO) ?
		V4L2_TUNER_MODE_MONO : V4L2_TUNER_MODE_STEREO;
304
	v->signal = tea->tuned ? 0xffff : 0;
305 306 307 308
	return 0;
}

static int vidioc_s_tuner(struct file *file, void *priv,
309
					const struct v4l2_tuner *v)
310
{
311
	struct snd_tea575x *tea = video_drvdata(file);
312
	u32 orig_val = tea->val;
313 314

	if (v->index)
315
		return -EINVAL;
316 317 318
	tea->val &= ~TEA575X_BIT_MONO;
	if (v->audmode == V4L2_TUNER_MODE_MONO)
		tea->val |= TEA575X_BIT_MONO;
319 320 321 322
	/* Only apply changes if currently tuning FM */
	if (tea->band != BAND_AM && tea->val != orig_val)
		snd_tea575x_set_freq(tea);

323 324 325 326 327 328 329 330
	return 0;
}

static int vidioc_g_frequency(struct file *file, void *priv,
					struct v4l2_frequency *f)
{
	struct snd_tea575x *tea = video_drvdata(file);

331 332
	if (f->tuner != 0)
		return -EINVAL;
333 334 335 336 337 338
	f->type = V4L2_TUNER_RADIO;
	f->frequency = tea->freq;
	return 0;
}

static int vidioc_s_frequency(struct file *file, void *priv,
339
					const struct v4l2_frequency *f)
340 341 342
{
	struct snd_tea575x *tea = video_drvdata(file);

343 344 345
	if (f->tuner != 0 || f->type != V4L2_TUNER_RADIO)
		return -EINVAL;

346 347 348 349 350 351 352
	if (tea->has_am && f->frequency < (20000 * 16))
		tea->band = BAND_AM;
	else if (tea->tea5759)
		tea->band = BAND_FM_JAPAN;
	else
		tea->band = BAND_FM;

353
	tea->freq = clamp_t(u32, f->frequency, bands[tea->band].rangelow,
354
					bands[tea->band].rangehigh);
355 356 357 358
	snd_tea575x_set_freq(tea);
	return 0;
}

359
static int vidioc_s_hw_freq_seek(struct file *file, void *fh,
360
					const struct v4l2_hw_freq_seek *a)
361
{
362
	struct snd_tea575x *tea = video_drvdata(file);
H
Hans Verkuil 已提交
363
	unsigned long timeout;
364
	int i, spacing;
365

366 367 368
	if (tea->cannot_read_data)
		return -ENOTTY;
	if (a->tuner || a->wrap_around)
369
		return -EINVAL;
H
Hans Verkuil 已提交
370

371 372 373
	if (file->f_flags & O_NONBLOCK)
		return -EWOULDBLOCK;

374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395
	if (a->rangelow || a->rangehigh) {
		for (i = 0; i < ARRAY_SIZE(bands); i++) {
			if ((i == BAND_FM && tea->tea5759) ||
			    (i == BAND_FM_JAPAN && !tea->tea5759) ||
			    (i == BAND_AM && !tea->has_am))
				continue;
			if (bands[i].rangelow  == a->rangelow &&
			    bands[i].rangehigh == a->rangehigh)
				break;
		}
		if (i == ARRAY_SIZE(bands))
			return -EINVAL; /* No matching band found */
		if (i != tea->band) {
			tea->band = i;
			tea->freq = clamp(tea->freq, bands[i].rangelow,
						     bands[i].rangehigh);
			snd_tea575x_set_freq(tea);
		}
	}

	spacing = (tea->band == BAND_AM) ? 5 : 50; /* kHz */

H
Hans Verkuil 已提交
396 397
	/* clear the frequency, HW will fill it in */
	tea->val &= ~TEA575X_BIT_FREQ_MASK;
398 399 400
	tea->val |= TEA575X_BIT_SEARCH;
	if (a->seek_upward)
		tea->val |= TEA575X_BIT_UPDOWN;
H
Hans Verkuil 已提交
401 402
	else
		tea->val &= ~TEA575X_BIT_UPDOWN;
403
	snd_tea575x_write(tea, tea->val);
H
Hans Verkuil 已提交
404
	timeout = jiffies + msecs_to_jiffies(10000);
405
	for (;;) {
H
Hans Verkuil 已提交
406 407
		if (time_after(jiffies, timeout))
			break;
408 409 410
		if (schedule_timeout_interruptible(msecs_to_jiffies(10))) {
			/* some signal arrived, stop search */
			tea->val &= ~TEA575X_BIT_SEARCH;
H
Hans Verkuil 已提交
411
			snd_tea575x_set_freq(tea);
412 413
			return -ERESTARTSYS;
		}
H
Hans Verkuil 已提交
414 415 416 417 418 419 420 421 422 423 424 425 426
		if (!(snd_tea575x_read(tea) & TEA575X_BIT_SEARCH)) {
			u32 freq;

			/* Found a frequency, wait until it can be read */
			for (i = 0; i < 100; i++) {
				msleep(10);
				freq = snd_tea575x_get_freq(tea);
				if (freq) /* available */
					break;
			}
			if (freq == 0) /* shouldn't happen */
				break;
			/*
427 428
			 * if we moved by less than the spacing, or in the
			 * wrong direction, continue seeking
H
Hans Verkuil 已提交
429
			 */
430
			if (abs(tea->freq - freq) < 16 * spacing ||
H
Hans Verkuil 已提交
431 432 433 434 435 436 437 438 439
					(a->seek_upward && freq < tea->freq) ||
					(!a->seek_upward && freq > tea->freq)) {
				snd_tea575x_write(tea, tea->val);
				continue;
			}
			tea->freq = freq;
			tea->val &= ~TEA575X_BIT_SEARCH;
			return 0;
		}
440
	}
H
Hans Verkuil 已提交
441 442
	tea->val &= ~TEA575X_BIT_SEARCH;
	snd_tea575x_set_freq(tea);
443
	return -ENODATA;
444 445
}

446
static int tea575x_s_ctrl(struct v4l2_ctrl *ctrl)
447
{
448
	struct snd_tea575x *tea = container_of(ctrl->handler, struct snd_tea575x, ctrl_handler);
449 450 451

	switch (ctrl->id) {
	case V4L2_CID_AUDIO_MUTE:
452 453
		tea->mute = ctrl->val;
		snd_tea575x_set_freq(tea);
454
		return 0;
455 456 457 458 459 460
	}

	return -EINVAL;
}

static const struct v4l2_file_operations tea575x_fops = {
461
	.unlocked_ioctl	= video_ioctl2,
462 463 464
	.open           = v4l2_fh_open,
	.release        = v4l2_fh_release,
	.poll           = v4l2_ctrl_poll,
465 466 467 468 469 470 471 472
};

static const struct v4l2_ioctl_ops tea575x_ioctl_ops = {
	.vidioc_querycap    = vidioc_querycap,
	.vidioc_g_tuner     = vidioc_g_tuner,
	.vidioc_s_tuner     = vidioc_s_tuner,
	.vidioc_g_frequency = vidioc_g_frequency,
	.vidioc_s_frequency = vidioc_s_frequency,
473
	.vidioc_s_hw_freq_seek = vidioc_s_hw_freq_seek,
474
	.vidioc_enum_freq_bands = vidioc_enum_freq_bands,
475 476 477
	.vidioc_log_status  = v4l2_ctrl_log_status,
	.vidioc_subscribe_event = v4l2_ctrl_subscribe_event,
	.vidioc_unsubscribe_event = v4l2_event_unsubscribe,
478 479
};

480
static const struct video_device tea575x_radio = {
481
	.ioctl_ops 	= &tea575x_ioctl_ops,
482
	.release        = video_device_release_empty,
483 484 485 486
};

static const struct v4l2_ctrl_ops tea575x_ctrl_ops = {
	.s_ctrl = tea575x_s_ctrl,
487 488
};

L
Linus Torvalds 已提交
489

490 491
int snd_tea575x_hw_init(struct snd_tea575x *tea)
{
492
	tea->mute = true;
493

494 495 496 497 498 499 500
	/* Not all devices can or know how to read the data back.
	   Such devices can set cannot_read_data to true. */
	if (!tea->cannot_read_data) {
		snd_tea575x_write(tea, 0x55AA);
		if (snd_tea575x_read(tea) != 0x55AA)
			return -ENODEV;
	}
L
Linus Torvalds 已提交
501

H
Hans Verkuil 已提交
502
	tea->val = TEA575X_BIT_BAND_FM | TEA575X_BIT_SEARCH_5_28;
503
	tea->freq = 90500 * 16;		/* 90.5Mhz default */
504
	snd_tea575x_set_freq(tea);
505

506 507 508 509 510 511 512 513 514 515 516
	return 0;
}
EXPORT_SYMBOL(snd_tea575x_hw_init);

int snd_tea575x_init(struct snd_tea575x *tea, struct module *owner)
{
	int retval = snd_tea575x_hw_init(tea);

	if (retval)
		return retval;

517 518
	tea->vd = tea575x_radio;
	video_set_drvdata(&tea->vd, tea);
519
	mutex_init(&tea->mutex);
520
	strlcpy(tea->vd.name, tea->v4l2_dev->name, sizeof(tea->vd.name));
521
	tea->vd.lock = &tea->mutex;
522
	tea->vd.v4l2_dev = tea->v4l2_dev;
523 524 525
	tea->fops = tea575x_fops;
	tea->fops.owner = owner;
	tea->vd.fops = &tea->fops;
526
	set_bit(V4L2_FL_USE_FH_PRIO, &tea->vd.flags);
527 528
	/* disable hw_freq_seek if we can't use it */
	if (tea->cannot_read_data)
529
		v4l2_disable_ioctl(&tea->vd, VIDIOC_S_HW_FREQ_SEEK);
L
Linus Torvalds 已提交
530

531 532 533 534 535 536
	if (!tea->cannot_mute) {
		tea->vd.ctrl_handler = &tea->ctrl_handler;
		v4l2_ctrl_handler_init(&tea->ctrl_handler, 1);
		v4l2_ctrl_new_std(&tea->ctrl_handler, &tea575x_ctrl_ops,
				  V4L2_CID_AUDIO_MUTE, 0, 1, 1, 1);
		retval = tea->ctrl_handler.error;
537
		if (retval) {
538
			v4l2_err(tea->v4l2_dev, "can't initialize controls\n");
539 540 541
			v4l2_ctrl_handler_free(&tea->ctrl_handler);
			return retval;
		}
542

543 544 545 546 547 548 549 550 551 552
		if (tea->ext_init) {
			retval = tea->ext_init(tea);
			if (retval) {
				v4l2_ctrl_handler_free(&tea->ctrl_handler);
				return retval;
			}
		}

		v4l2_ctrl_handler_setup(&tea->ctrl_handler);
	}
553

554
	retval = video_register_device(&tea->vd, VFL_TYPE_RADIO, tea->radio_nr);
555
	if (retval) {
556
		v4l2_err(tea->v4l2_dev, "can't register video device!\n");
557
		v4l2_ctrl_handler_free(tea->vd.ctrl_handler);
558
		return retval;
559
	}
L
Linus Torvalds 已提交
560

561
	return 0;
L
Linus Torvalds 已提交
562 563
}

564
void snd_tea575x_exit(struct snd_tea575x *tea)
L
Linus Torvalds 已提交
565
{
566
	video_unregister_device(&tea->vd);
567
	v4l2_ctrl_handler_free(tea->vd.ctrl_handler);
L
Linus Torvalds 已提交
568 569 570 571 572 573
}

static int __init alsa_tea575x_module_init(void)
{
	return 0;
}
574

L
Linus Torvalds 已提交
575 576 577
static void __exit alsa_tea575x_module_exit(void)
{
}
578

L
Linus Torvalds 已提交
579 580 581 582 583
module_init(alsa_tea575x_module_init)
module_exit(alsa_tea575x_module_exit)

EXPORT_SYMBOL(snd_tea575x_init);
EXPORT_SYMBOL(snd_tea575x_exit);
584
EXPORT_SYMBOL(snd_tea575x_set_freq);