Anforderungen  |   Konzepte  |   Entwurf  |   Entwicklung  |   Qualitätssicherung  |   Lebenszyklus  |   Steuerung
 
 
 
 


Quelle  ce6230.c   Sprache: C

 
// SPDX-License-Identifier: GPL-2.0-or-later
/*
 * Intel CE6230 DVB USB driver
 *
 * Copyright (C) 2009 Antti Palosaari <crope@iki.fi>
 */


#include "ce6230.h"

DVB_DEFINE_MOD_OPT_ADAPTER_NR(adapter_nr);

static int ce6230_ctrl_msg(struct dvb_usb_device *d, struct usb_req *req)
{
 int ret;
 unsigned int pipe;
 u8 request;
 u8 requesttype;
 u16 value;
 u16 index;
 u8 *buf;

 request = req->cmd;
 value = req->value;
 index = req->index;

 switch (req->cmd) {
 case I2C_READ:
 case DEMOD_READ:
 case REG_READ:
  requesttype = (USB_TYPE_VENDOR | USB_DIR_IN);
  break;
 case I2C_WRITE:
 case DEMOD_WRITE:
 case REG_WRITE:
  requesttype = (USB_TYPE_VENDOR | USB_DIR_OUT);
  break;
 default:
  dev_err(&d->udev->dev, "%s: unknown command=%02x\n",
    KBUILD_MODNAME, req->cmd);
  ret = -EINVAL;
  goto error;
 }

 buf = kmalloc(req->data_len, GFP_KERNEL);
 if (!buf) {
  ret = -ENOMEM;
  goto error;
 }

 if (requesttype == (USB_TYPE_VENDOR | USB_DIR_OUT)) {
  /* write */
  memcpy(buf, req->data, req->data_len);
  pipe = usb_sndctrlpipe(d->udev, 0);
 } else {
  /* read */
  pipe = usb_rcvctrlpipe(d->udev, 0);
 }

 msleep(1); /* avoid I2C errors */

 ret = usb_control_msg(d->udev, pipe, request, requesttype, value, index,
   buf, req->data_len, CE6230_USB_TIMEOUT);

 dvb_usb_dbg_usb_control_msg(d->udev, request, requesttype, value, index,
   buf, req->data_len);

 if (ret < 0)
  dev_err(&d->udev->dev, "%s: usb_control_msg() failed=%d\n",
    KBUILD_MODNAME, ret);
 else
  ret = 0;

 /* read request, copy returned data to return buf */
 if (!ret && requesttype == (USB_TYPE_VENDOR | USB_DIR_IN))
  memcpy(req->data, buf, req->data_len);

 kfree(buf);
error:
 return ret;
}

/* I2C */
static struct zl10353_config ce6230_zl10353_config;

static int ce6230_i2c_master_xfer(struct i2c_adapter *adap,
  struct i2c_msg msg[], int num)
{
 struct dvb_usb_device *d = i2c_get_adapdata(adap);
 int ret = 0, i = 0;
 struct usb_req req;

 if (num > 2)
  return -EOPNOTSUPP;

 memset(&req, 0, sizeof(req));

 if (mutex_lock_interruptible(&d->i2c_mutex) < 0)
  return -EAGAIN;

 while (i < num) {
  if (num > i + 1 && (msg[i+1].flags & I2C_M_RD)) {
   if (msg[i].addr ==
    ce6230_zl10353_config.demod_address) {
    if (msg[i].len < 1) {
     i = -EOPNOTSUPP;
     break;
    }
    req.cmd = DEMOD_READ;
    req.value = msg[i].addr >> 1;
    req.index = msg[i].buf[0];
    req.data_len = msg[i+1].len;
    req.data = &msg[i+1].buf[0];
    ret = ce6230_ctrl_msg(d, &req);
   } else {
    dev_err(&d->udev->dev, "%s: I2C read not " \
      "implemented\n",
      KBUILD_MODNAME);
    ret = -EOPNOTSUPP;
   }
   i += 2;
  } else {
   if (msg[i].addr ==
    ce6230_zl10353_config.demod_address) {
    if (msg[i].len < 1) {
     i = -EOPNOTSUPP;
     break;
    }
    req.cmd = DEMOD_WRITE;
    req.value = msg[i].addr >> 1;
    req.index = msg[i].buf[0];
    req.data_len = msg[i].len-1;
    req.data = &msg[i].buf[1];
    ret = ce6230_ctrl_msg(d, &req);
   } else {
    req.cmd = I2C_WRITE;
    req.value = 0x2000 + (msg[i].addr >> 1);
    req.index = 0x0000;
    req.data_len = msg[i].len;
    req.data = &msg[i].buf[0];
    ret = ce6230_ctrl_msg(d, &req);
   }
   i += 1;
  }
  if (ret)
   break;
 }

 mutex_unlock(&d->i2c_mutex);
 return ret ? ret : i;
}

static u32 ce6230_i2c_functionality(struct i2c_adapter *adapter)
{
 return I2C_FUNC_I2C;
}

static const struct i2c_algorithm ce6230_i2c_algorithm = {
 .master_xfer   = ce6230_i2c_master_xfer,
 .functionality = ce6230_i2c_functionality,
};

/* Callbacks for DVB USB */
static struct zl10353_config ce6230_zl10353_config = {
 .demod_address = 0x1e,
 .adc_clock = 450000,
 .if2 = 45700,
 .no_tuner = 1,
 .parallel_ts = 1,
 .clock_ctl_1 = 0x34,
 .pll_0 = 0x0e,
};

static int ce6230_zl10353_frontend_attach(struct dvb_usb_adapter *adap)
{
 struct dvb_usb_device *d = adap_to_d(adap);

 dev_dbg(&d->udev->dev, "%s:\n", __func__);

 adap->fe[0] = dvb_attach(zl10353_attach, &ce6230_zl10353_config,
   &d->i2c_adap);
 if (adap->fe[0] == NULL)
  return -ENODEV;

 return 0;
}

static struct mxl5005s_config ce6230_mxl5003s_config = {
 .i2c_address     = 0xc6,
 .if_freq         = IF_FREQ_4570000HZ,
 .xtal_freq       = CRYSTAL_FREQ_16000000HZ,
 .agc_mode        = MXL_SINGLE_AGC,
 .tracking_filter = MXL_TF_DEFAULT,
 .rssi_enable     = MXL_RSSI_ENABLE,
 .cap_select      = MXL_CAP_SEL_ENABLE,
 .div_out         = MXL_DIV_OUT_4,
 .clock_out       = MXL_CLOCK_OUT_DISABLE,
 .output_load     = MXL5005S_IF_OUTPUT_LOAD_200_OHM,
 .top   = MXL5005S_TOP_25P2,
 .mod_mode        = MXL_DIGITAL_MODE,
 .if_mode         = MXL_ZERO_IF,
 .AgcMasterByte   = 0x00,
};

static int ce6230_mxl5003s_tuner_attach(struct dvb_usb_adapter *adap)
{
 struct dvb_usb_device *d = adap_to_d(adap);
 int ret;

 dev_dbg(&d->udev->dev, "%s:\n", __func__);

 ret = dvb_attach(mxl5005s_attach, adap->fe[0], &d->i2c_adap,
   &ce6230_mxl5003s_config) == NULL ? -ENODEV : 0;
 return ret;
}

static int ce6230_power_ctrl(struct dvb_usb_device *d, int onoff)
{
 int ret;

 dev_dbg(&d->udev->dev, "%s: onoff=%d\n", __func__, onoff);

 /* InterfaceNumber 1 / AlternateSetting 0     idle
   InterfaceNumber 1 / AlternateSetting 1     streaming */

 ret = usb_set_interface(d->udev, 1, onoff);
 if (ret)
  dev_err(&d->udev->dev, "%s: usb_set_interface() failed=%d\n",
    KBUILD_MODNAME, ret);

 return ret;
}

/* DVB USB Driver stuff */
static struct dvb_usb_device_properties ce6230_props = {
 .driver_name = KBUILD_MODNAME,
 .owner = THIS_MODULE,
 .adapter_nr = adapter_nr,
 .bInterfaceNumber = 1,

 .i2c_algo = &ce6230_i2c_algorithm,
 .power_ctrl = ce6230_power_ctrl,
 .frontend_attach = ce6230_zl10353_frontend_attach,
 .tuner_attach = ce6230_mxl5003s_tuner_attach,

 .num_adapters = 1,
 .adapter = {
  {
   .stream = {
    .type = USB_BULK,
    .count = 6,
    .endpoint = 0x82,
    .u = {
     .bulk = {
      .buffersize = (16 * 512),
     }
    }
   },
  }
 },
};

static const struct usb_device_id ce6230_id_table[] = {
 { DVB_USB_DEVICE(USB_VID_INTEL, USB_PID_INTEL_CE9500,
  &ce6230_props, "Intel CE9500 reference design", NULL) },
 { DVB_USB_DEVICE(USB_VID_AVERMEDIA, USB_PID_AVERMEDIA_A310,
  &ce6230_props, "AVerMedia A310 USB 2.0 DVB-T tuner", NULL) },
 { }
};
MODULE_DEVICE_TABLE(usb, ce6230_id_table);

static struct usb_driver ce6230_usb_driver = {
 .name = KBUILD_MODNAME,
 .id_table = ce6230_id_table,
 .probe = dvb_usbv2_probe,
 .disconnect = dvb_usbv2_disconnect,
 .suspend = dvb_usbv2_suspend,
 .resume = dvb_usbv2_resume,
 .reset_resume = dvb_usbv2_reset_resume,
 .no_dynamic_id = 1,
 .soft_unbind = 1,
};

module_usb_driver(ce6230_usb_driver);

MODULE_AUTHOR("Antti Palosaari ");
MODULE_DESCRIPTION("Intel CE6230 driver");
MODULE_LICENSE("GPL");

Messung V0.5
C=96 H=94 G=94

¤ Dauer der Verarbeitung: 0.3 Sekunden  ¤

*© Formatika GbR, Deutschland






Wurzel

Suchen

Beweissystem der NASA

Beweissystem Isabelle

NIST Cobol Testsuite

Cephes Mathematical Library

Wiener Entwicklungsmethode

Haftungshinweis

Die Informationen auf dieser Webseite wurden nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit, noch Qualität der bereit gestellten Informationen zugesichert.

Bemerkung:

Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.






                                                                                                                                                                                                                                                                                                                                                                                                     


Neuigkeiten

     Aktuelles
     Motto des Tages

Software

     Produkte
     Quellcodebibliothek

Aktivitäten

     Artikel über Sicherheit
     Anleitung zur Aktivierung von SSL

Muße

     Gedichte
     Musik
     Bilder

Jenseits des Üblichen ....
    

Besucherstatistik

Besucherstatistik

Monitoring

Montastic status badge