⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 m920x.c

📁 linux内核源码
💻 C
📖 第 1 页 / 共 2 页
字号:
/* DVB USB compliant linux driver for MSI Mega Sky 580 DVB-T USB2.0 receiver * * Copyright (C) 2006 Aapo Tahkola (aet@rasterburn.org) * *	This program is free software; you can redistribute it and/or modify it *	under the terms of the GNU General Public License as published by the *	Free Software Foundation, version 2. * * see Documentation/dvb/README.dvb-usb for more information */#include "m920x.h"#include "mt352.h"#include "mt352_priv.h"#include "qt1010.h"#include "tda1004x.h"#include "tda827x.h"/* debug */static int dvb_usb_m920x_debug;module_param_named(debug,dvb_usb_m920x_debug, int, 0644);MODULE_PARM_DESC(debug, "set debugging level (1=rc (or-able))." DVB_USB_DEBUG_STATUS);static int m920x_set_filter(struct dvb_usb_device *d, int type, int idx, int pid);static inline int m920x_read(struct usb_device *udev, u8 request, u16 value,			     u16 index, void *data, int size){	int ret;	ret = usb_control_msg(udev, usb_rcvctrlpipe(udev, 0),			      request, USB_TYPE_VENDOR | USB_DIR_IN,			      value, index, data, size, 2000);	if (ret < 0) {		printk(KERN_INFO "m920x_read = error: %d\n", ret);		return ret;	}	if (ret != size) {		deb("m920x_read = no data\n");		return -EIO;	}	return 0;}static inline int m920x_write(struct usb_device *udev, u8 request,			      u16 value, u16 index){	int ret;	ret = usb_control_msg(udev, usb_sndctrlpipe(udev, 0),			      request, USB_TYPE_VENDOR | USB_DIR_OUT,			      value, index, NULL, 0, 2000);	return ret;}static int m920x_init(struct dvb_usb_device *d, struct m920x_inits *rc_seq){	int ret = 0, i, epi, flags = 0;	int adap_enabled[M9206_MAX_ADAPTERS] = { 0 };	/* Remote controller init. */	if (d->props.rc_query) {		deb("Initialising remote control\n");		while (rc_seq->address) {			if ((ret = m920x_write(d->udev, M9206_CORE,					       rc_seq->data,					       rc_seq->address)) != 0) {				deb("Initialising remote control failed\n");				return ret;			}			rc_seq++;		}		deb("Initialising remote control success\n");	}	for (i = 0; i < d->props.num_adapters; i++)		flags |= d->adapter[i].props.caps;	/* Some devices(Dposh) might crash if we attempt touch at all. */	if (flags & DVB_USB_ADAP_HAS_PID_FILTER) {		for (i = 0; i < d->props.num_adapters; i++) {			epi = d->adapter[i].props.stream.endpoint - 0x81;			if (epi < 0 || epi >= M9206_MAX_ADAPTERS) {				printk(KERN_INFO "m920x: Unexpected adapter endpoint!\n");				return -EINVAL;			}			adap_enabled[epi] = 1;		}		for (i = 0; i < M9206_MAX_ADAPTERS; i++) {			if (adap_enabled[i])				continue;			if ((ret = m920x_set_filter(d, 0x81 + i, 0, 0x0)) != 0)				return ret;			if ((ret = m920x_set_filter(d, 0x81 + i, 0, 0x02f5)) != 0)				return ret;		}	}	return ret;}static int m920x_init_ep(struct usb_interface *intf){	struct usb_device *udev = interface_to_usbdev(intf);	struct usb_host_interface *alt;	if ((alt = usb_altnum_to_altsetting(intf, 1)) == NULL) {		deb("No alt found!\n");		return -ENODEV;	}	return usb_set_interface(udev, alt->desc.bInterfaceNumber,				 alt->desc.bAlternateSetting);}static int m920x_rc_query(struct dvb_usb_device *d, u32 *event, int *state){	struct m920x_state *m = d->priv;	int i, ret = 0;	u8 rc_state[2];	if ((ret = m920x_read(d->udev, M9206_CORE, 0x0, M9206_RC_STATE, rc_state, 1)) != 0)		goto unlock;	if ((ret = m920x_read(d->udev, M9206_CORE, 0x0, M9206_RC_KEY, rc_state + 1, 1)) != 0)		goto unlock;	for (i = 0; i < d->props.rc_key_map_size; i++)		if (d->props.rc_key_map[i].data == rc_state[1]) {			*event = d->props.rc_key_map[i].event;			switch(rc_state[0]) {			case 0x80:				*state = REMOTE_NO_KEY_PRESSED;				goto unlock;			case 0x88: /* framing error or "invalid code" */			case 0x99:			case 0xc0:			case 0xd8:				*state = REMOTE_NO_KEY_PRESSED;				m->rep_count = 0;				goto unlock;			case 0x93:			case 0x92:				m->rep_count = 0;				*state = REMOTE_KEY_PRESSED;				goto unlock;			case 0x91:				/* prevent immediate auto-repeat */				if (++m->rep_count > 2)					*state = REMOTE_KEY_REPEAT;				else					*state = REMOTE_NO_KEY_PRESSED;				goto unlock;			default:				deb("Unexpected rc state %02x\n", rc_state[0]);				*state = REMOTE_NO_KEY_PRESSED;				goto unlock;			}		}	if (rc_state[1] != 0)		deb("Unknown rc key %02x\n", rc_state[1]);	*state = REMOTE_NO_KEY_PRESSED; unlock:	return ret;}/* I2C */static int m920x_i2c_xfer(struct i2c_adapter *adap, struct i2c_msg msg[], int num){	struct dvb_usb_device *d = i2c_get_adapdata(adap);	int i, j;	int ret = 0;	if (!num)		return -EINVAL;	if (mutex_lock_interruptible(&d->i2c_mutex) < 0)		return -EAGAIN;	for (i = 0; i < num; i++) {		if (msg[i].flags & (I2C_M_NO_RD_ACK | I2C_M_IGNORE_NAK | I2C_M_TEN) || msg[i].len == 0) {			/* For a 0 byte message, I think sending the address			 * to index 0x80|0x40 would be the correct thing to			 * do.  However, zero byte messages are only used for			 * probing, and since we don't know how to get the			 * slave's ack, we can't probe. */			ret = -ENOTSUPP;			goto unlock;		}		/* Send START & address/RW bit */		if (!(msg[i].flags & I2C_M_NOSTART)) {			if ((ret = m920x_write(d->udev, M9206_I2C,					(msg[i].addr << 1) |					(msg[i].flags & I2C_M_RD ? 0x01 : 0), 0x80)) != 0)				goto unlock;			/* Should check for ack here, if we knew how. */		}		if (msg[i].flags & I2C_M_RD) {			for (j = 0; j < msg[i].len; j++) {				/* Last byte of transaction?				 * Send STOP, otherwise send ACK. */				int stop = (i+1 == num && j+1 == msg[i].len) ? 0x40 : 0x01;				if ((ret = m920x_read(d->udev, M9206_I2C, 0x0,						      0x20 | stop,						      &msg[i].buf[j], 1)) != 0)					goto unlock;			}		} else {			for (j = 0; j < msg[i].len; j++) {				/* Last byte of transaction? Then send STOP. */				int stop = (i+1 == num && j+1 == msg[i].len) ? 0x40 : 0x00;				if ((ret = m920x_write(d->udev, M9206_I2C, msg[i].buf[j], stop)) != 0)					goto unlock;				/* Should check for ack here too. */			}		}	}	ret = num; unlock:	mutex_unlock(&d->i2c_mutex);	return ret;}static u32 m920x_i2c_func(struct i2c_adapter *adapter){	return I2C_FUNC_I2C;}static struct i2c_algorithm m920x_i2c_algo = {	.master_xfer   = m920x_i2c_xfer,	.functionality = m920x_i2c_func,};/* pid filter */static int m920x_set_filter(struct dvb_usb_device *d, int type, int idx, int pid){	int ret = 0;	if (pid >= 0x8000)		return -EINVAL;	pid |= 0x8000;	if ((ret = m920x_write(d->udev, M9206_FILTER, pid, (type << 8) | (idx * 4) )) != 0)		return ret;	if ((ret = m920x_write(d->udev, M9206_FILTER, 0, (type << 8) | (idx * 4) )) != 0)		return ret;	return ret;}static int m920x_update_filters(struct dvb_usb_adapter *adap){	struct m920x_state *m = adap->dev->priv;	int enabled = m->filtering_enabled[adap->id];	int i, ret = 0, filter = 0;	int ep = adap->props.stream.endpoint;	for (i = 0; i < M9206_MAX_FILTERS; i++)		if (m->filters[adap->id][i] == 8192)			enabled = 0;	/* Disable all filters */	if ((ret = m920x_set_filter(adap->dev, ep, 1, enabled)) != 0)		return ret;	for (i = 0; i < M9206_MAX_FILTERS; i++)		if ((ret = m920x_set_filter(adap->dev, ep, i + 2, 0)) != 0)			return ret;	/* Set */	if (enabled) {		for (i = 0; i < M9206_MAX_FILTERS; i++) {			if (m->filters[adap->id][i] == 0)				continue;			if ((ret = m920x_set_filter(adap->dev, ep, filter + 2, m->filters[adap->id][i])) != 0)				return ret;			filter++;		}	}	return ret;}static int m920x_pid_filter_ctrl(struct dvb_usb_adapter *adap, int onoff){	struct m920x_state *m = adap->dev->priv;	m->filtering_enabled[adap->id] = onoff ? 1 : 0;	return m920x_update_filters(adap);}static int m920x_pid_filter(struct dvb_usb_adapter *adap, int index, u16 pid, int onoff){	struct m920x_state *m = adap->dev->priv;	m->filters[adap->id][index] = onoff ? pid : 0;	return m920x_update_filters(adap);}static int m920x_firmware_download(struct usb_device *udev, const struct firmware *fw){	u16 value, index, size;	u8 read[4], *buff;	int i, pass, ret = 0;	buff = kmalloc(65536, GFP_KERNEL);	if ((ret = m920x_read(udev, M9206_FILTER, 0x0, 0x8000, read, 4)) != 0)		goto done;	deb("%x %x %x %x\n", read[0], read[1], read[2], read[3]);	if ((ret = m920x_read(udev, M9206_FW, 0x0, 0x0, read, 1)) != 0)		goto done;	deb("%x\n", read[0]);	for (pass = 0; pass < 2; pass++) {		for (i = 0; i + (sizeof(u16) * 3) < fw->size;) {			value = le16_to_cpu(*(u16 *)(fw->data + i));			i += sizeof(u16);			index = le16_to_cpu(*(u16 *)(fw->data + i));			i += sizeof(u16);			size = le16_to_cpu(*(u16 *)(fw->data + i));			i += sizeof(u16);			if (pass == 1) {				/* Will stall if using fw->data ... */				memcpy(buff, fw->data + i, size);				ret = usb_control_msg(udev, usb_sndctrlpipe(udev,0),						      M9206_FW,						      USB_TYPE_VENDOR | USB_DIR_OUT,						      value, index, buff, size, 20);				if (ret != size) {					deb("error while uploading fw!\n");					ret = -EIO;					goto done;				}				msleep(3);			}			i += size;		}		if (i != fw->size) {			deb("bad firmware file!\n");			ret = -EINVAL;			goto done;		}	}	msleep(36);	/* m920x will disconnect itself from the bus after this. */	(void) m920x_write(udev, M9206_CORE, 0x01, M9206_FW_GO);	deb("firmware uploaded!\n"); done:	kfree(buff);	return ret;}/* Callbacks for DVB USB */static int m920x_identify_state(struct usb_device *udev,				struct dvb_usb_device_properties *props,				struct dvb_usb_device_description **desc,				int *cold){	struct usb_host_interface *alt;	alt = usb_altnum_to_altsetting(usb_ifnum_to_if(udev, 0), 1);	*cold = (alt == NULL) ? 1 : 0;	return 0;}/* demod configurations */static int m920x_mt352_demod_init(struct dvb_frontend *fe){	int ret;	u8 config[] = { CONFIG, 0x3d };	u8 clock[] = { CLOCK_CTL, 0x30 };	u8 reset[] = { RESET, 0x80 };	u8 adc_ctl[] = { ADC_CTL_1, 0x40 };	u8 agc[] = { AGC_TARGET, 0x1c, 0x20 };	u8 sec_agc[] = { 0x69, 0x00, 0xff, 0xff, 0x40, 0xff, 0x00, 0x40, 0x40 };	u8 unk1[] = { 0x93, 0x1a };	u8 unk2[] = { 0xb5, 0x7a };	deb("Demod init!\n");	if ((ret = mt352_write(fe, config, ARRAY_SIZE(config))) != 0)		return ret;	if ((ret = mt352_write(fe, clock, ARRAY_SIZE(clock))) != 0)		return ret;	if ((ret = mt352_write(fe, reset, ARRAY_SIZE(reset))) != 0)		return ret;	if ((ret = mt352_write(fe, adc_ctl, ARRAY_SIZE(adc_ctl))) != 0)		return ret;	if ((ret = mt352_write(fe, agc, ARRAY_SIZE(agc))) != 0)		return ret;	if ((ret = mt352_write(fe, sec_agc, ARRAY_SIZE(sec_agc))) != 0)		return ret;	if ((ret = mt352_write(fe, unk1, ARRAY_SIZE(unk1))) != 0)		return ret;	if ((ret = mt352_write(fe, unk2, ARRAY_SIZE(unk2))) != 0)		return ret;	return 0;}static struct mt352_config m920x_mt352_config = {	.demod_address = 0x0f,	.no_tuner = 1,	.demod_init = m920x_mt352_demod_init,};static struct tda1004x_config m920x_tda10046_08_config = {	.demod_address = 0x08,	.invert = 0,	.invert_oclk = 0,	.ts_mode = TDA10046_TS_SERIAL,	.xtal_freq = TDA10046_XTAL_16M,	.if_freq = TDA10046_FREQ_045,	.agc_config = TDA10046_AGC_TDA827X,	.gpio_config = TDA10046_GPTRI,	.request_firmware = NULL,};static struct tda1004x_config m920x_tda10046_0b_config = {	.demod_address = 0x0b,	.invert = 0,	.invert_oclk = 0,	.ts_mode = TDA10046_TS_SERIAL,

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -