blob: e1e9bddcf516a2a2476f3600be171097f08dbf4d [file] [log] [blame]
Antti Palosaari85bc9b52011-08-01 00:52:11 -03001/*
2 * Allegro A8293 SEC driver
3 *
4 * Copyright (C) 2011 Antti Palosaari <crope@iki.fi>
5 *
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation; either version 2 of the License, or
9 * (at your option) any later version.
10 *
11 * This program is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
Antti Palosaari85bc9b52011-08-01 00:52:11 -030015 */
16
Antti Palosaari85bc9b52011-08-01 00:52:11 -030017#include "a8293.h"
18
Antti Palosaari2c509b82015-04-20 18:16:19 -030019struct a8293_dev {
Antti Palosaarib561bae2015-04-16 13:28:39 -030020 struct i2c_client *client;
Antti Palosaari85bc9b52011-08-01 00:52:11 -030021 u8 reg[2];
22};
23
Antti Palosaari85bc9b52011-08-01 00:52:11 -030024static int a8293_set_voltage(struct dvb_frontend *fe,
Antti Palosaari3250a5502015-04-20 18:57:03 -030025 enum fe_sec_voltage fe_sec_voltage)
Antti Palosaari85bc9b52011-08-01 00:52:11 -030026{
Antti Palosaari2c509b82015-04-20 18:16:19 -030027 struct a8293_dev *dev = fe->sec_priv;
28 struct i2c_client *client = dev->client;
Antti Palosaari85bc9b52011-08-01 00:52:11 -030029 int ret;
Antti Palosaari4bef67e2015-04-20 18:47:44 -030030 u8 reg0, reg1;
Antti Palosaari85bc9b52011-08-01 00:52:11 -030031
Antti Palosaari2c509b82015-04-20 18:16:19 -030032 dev_dbg(&client->dev, "fe_sec_voltage=%d\n", fe_sec_voltage);
Antti Palosaari85bc9b52011-08-01 00:52:11 -030033
34 switch (fe_sec_voltage) {
35 case SEC_VOLTAGE_OFF:
36 /* ENB=0 */
Antti Palosaari4bef67e2015-04-20 18:47:44 -030037 reg0 = 0x10;
Antti Palosaari85bc9b52011-08-01 00:52:11 -030038 break;
39 case SEC_VOLTAGE_13:
40 /* VSEL0=1, VSEL1=0, VSEL2=0, VSEL3=0, ENB=1*/
Antti Palosaari4bef67e2015-04-20 18:47:44 -030041 reg0 = 0x31;
Antti Palosaari85bc9b52011-08-01 00:52:11 -030042 break;
43 case SEC_VOLTAGE_18:
44 /* VSEL0=0, VSEL1=0, VSEL2=0, VSEL3=1, ENB=1*/
Antti Palosaari4bef67e2015-04-20 18:47:44 -030045 reg0 = 0x38;
Antti Palosaari85bc9b52011-08-01 00:52:11 -030046 break;
47 default:
48 ret = -EINVAL;
49 goto err;
Peter Senna Tschudinc2c1b412012-09-28 05:37:22 -030050 }
Antti Palosaari4bef67e2015-04-20 18:47:44 -030051 if (reg0 != dev->reg[0]) {
52 ret = i2c_master_send(client, &reg0, 1);
53 if (ret < 0)
54 goto err;
55 dev->reg[0] = reg0;
56 }
Antti Palosaari85bc9b52011-08-01 00:52:11 -030057
Antti Palosaari4bef67e2015-04-20 18:47:44 -030058 /* TMODE=0, TGATE=1 */
59 reg1 = 0x82;
60 if (reg1 != dev->reg[1]) {
61 ret = i2c_master_send(client, &reg1, 1);
62 if (ret < 0)
63 goto err;
64 dev->reg[1] = reg1;
65 }
Antti Palosaari85bc9b52011-08-01 00:52:11 -030066
Antti Palosaaric0ec1c42013-02-25 08:24:18 -030067 usleep_range(1500, 50000);
Antti Palosaari2c509b82015-04-20 18:16:19 -030068 return 0;
Antti Palosaari85bc9b52011-08-01 00:52:11 -030069err:
Antti Palosaari2c509b82015-04-20 18:16:19 -030070 dev_dbg(&client->dev, "failed=%d\n", ret);
Antti Palosaari85bc9b52011-08-01 00:52:11 -030071 return ret;
72}
73
Antti Palosaarib561bae2015-04-16 13:28:39 -030074static int a8293_probe(struct i2c_client *client,
Antti Palosaari55881b42015-04-20 17:39:33 -030075 const struct i2c_device_id *id)
Antti Palosaarib561bae2015-04-16 13:28:39 -030076{
Antti Palosaari2c509b82015-04-20 18:16:19 -030077 struct a8293_dev *dev;
Antti Palosaarib561bae2015-04-16 13:28:39 -030078 struct a8293_platform_data *pdata = client->dev.platform_data;
79 struct dvb_frontend *fe = pdata->dvb_frontend;
80 int ret;
81 u8 buf[2];
82
83 dev = kzalloc(sizeof(*dev), GFP_KERNEL);
84 if (!dev) {
85 ret = -ENOMEM;
86 goto err;
87 }
88
89 dev->client = client;
Antti Palosaarib561bae2015-04-16 13:28:39 -030090
91 /* check if the SEC is there */
Antti Palosaari2c509b82015-04-20 18:16:19 -030092 ret = i2c_master_recv(client, buf, 2);
93 if (ret < 0)
Antti Palosaarib561bae2015-04-16 13:28:39 -030094 goto err_kfree;
95
Antti Palosaarib561bae2015-04-16 13:28:39 -030096 /* override frontend ops */
97 fe->ops.set_voltage = a8293_set_voltage;
Antti Palosaarib561bae2015-04-16 13:28:39 -030098 fe->sec_priv = dev;
99 i2c_set_clientdata(client, dev);
100
101 dev_info(&client->dev, "Allegro A8293 SEC successfully attached\n");
102 return 0;
103err_kfree:
104 kfree(dev);
105err:
106 dev_dbg(&client->dev, "failed=%d\n", ret);
107 return ret;
108}
109
110static int a8293_remove(struct i2c_client *client)
111{
112 struct a8293_dev *dev = i2c_get_clientdata(client);
113
114 dev_dbg(&client->dev, "\n");
115
116 kfree(dev);
117 return 0;
118}
119
120static const struct i2c_device_id a8293_id_table[] = {
121 {"a8293", 0},
122 {}
123};
124MODULE_DEVICE_TABLE(i2c, a8293_id_table);
125
126static struct i2c_driver a8293_driver = {
127 .driver = {
Antti Palosaarib561bae2015-04-16 13:28:39 -0300128 .name = "a8293",
129 .suppress_bind_attrs = true,
130 },
131 .probe = a8293_probe,
132 .remove = a8293_remove,
133 .id_table = a8293_id_table,
134};
135
136module_i2c_driver(a8293_driver);
137
Antti Palosaari85bc9b52011-08-01 00:52:11 -0300138MODULE_AUTHOR("Antti Palosaari <crope@iki.fi>");
139MODULE_DESCRIPTION("Allegro A8293 SEC driver");
140MODULE_LICENSE("GPL");