aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/misc/cdc_tcxo.c
blob: d442e9550c87382b0668fb04e1a0efa09f1c36ed (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
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
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
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
157
158
159
160
161
162
163
164
165
166
167
168
/*
 * Clock Divider Chip (TCXO) support
 *
 * Copyright (C) 2010 Texas Instruments, Inc.
 * Written by Rajendra Nayak <rnayak@ti.com>
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License version 2 as
 * published by the Free Software Foundation.
 */

#include <linux/platform_device.h>
#include <linux/cdc_tcxo.h>
#include <linux/i2c.h>

#define DRIVER_DESC       "CDC TCXO driver"
#define DRIVER_NAME       "cdc_tcxo_driver"

struct cdc_tcxo_info {
	struct i2c_client *client;
	struct device *dev;
	char buf[4];
	int active;
};

static struct cdc_tcxo_info cdc_info;

int cdc_tcxo_set_req_int(int clk_id, int enable)
{
	char mask;
	int ret;

	if (!cdc_info.active)
		return -ENODEV;

	switch (clk_id) {
	case CDC_TCXO_CLK1:
		mask = CDC_TCXO_REQ1INT;
		break;
	case CDC_TCXO_CLK2:
		mask = CDC_TCXO_REQ2INT;
		break;
	case CDC_TCXO_CLK3:
		mask = CDC_TCXO_REQ3INT;
		break;
	case CDC_TCXO_CLK4:
		mask = CDC_TCXO_REQ4INT;
		break;
	default:
		dev_err(cdc_info.dev, "invalid clk_id: %d\n", clk_id);
		return -EINVAL;
	}

	if (enable)
		cdc_info.buf[CDC_TCXO_REG1] |= mask;
	else
		cdc_info.buf[CDC_TCXO_REG1] &= ~mask;

	ret = i2c_master_send(cdc_info.client, cdc_info.buf, CDC_TCXO_REGNUM);
	if (ret < 0) {
		dev_err(cdc_info.dev, "failed to write data (%d)\n", ret);
		return ret;
	}

	return 0;
}
EXPORT_SYMBOL(cdc_tcxo_set_req_int);

int cdc_tcxo_set_req_prio(int clk_id, int req_prio)
{
	char mask;
	int ret;

	if (!cdc_info.active)
		return -ENODEV;

	switch (clk_id) {
	case CDC_TCXO_CLK1:
		mask = CDC_TCXO_REQ1PRIO;
		break;
	case CDC_TCXO_CLK2:
		mask = CDC_TCXO_REQ2PRIO;
		break;
	case CDC_TCXO_CLK3:
		mask = CDC_TCXO_REQ3PRIO;
		break;
	case CDC_TCXO_CLK4:
		mask = CDC_TCXO_REQ4PRIO;
		break;
	default:
		dev_err(cdc_info.dev, "invalid clk_id: %d\n", clk_id);
		return -EINVAL;
	}

	if (req_prio == CDC_TCXO_PRIO_REQINT)
		cdc_info.buf[CDC_TCXO_REG3] |= mask;
	else
		cdc_info.buf[CDC_TCXO_REG3] &= ~mask;

	ret = i2c_master_send(cdc_info.client, cdc_info.buf, CDC_TCXO_REGNUM);
	if (ret < 0) {
		dev_err(cdc_info.dev, "failed to write data (%d)\n", ret);
		return ret;
	}

	return 0;
}
EXPORT_SYMBOL(cdc_tcxo_set_req_prio);

static int cdc_tcxo_probe(struct i2c_client *client, \
				const struct i2c_device_id *id)
{
	struct cdc_tcxo_platform_data *pdata = dev_get_platdata(&client->dev);
	int i, ret;

	cdc_info.client = client;
	cdc_info.dev = &client->dev;

	for (i = 0; i < CDC_TCXO_REGNUM; i++)
		cdc_info.buf[i] = pdata->buf[i];

	ret = i2c_master_send(client, cdc_info.buf, CDC_TCXO_REGNUM);
	if (ret < 0) {
		dev_err(cdc_info.dev, "failed to initialize registers (%d)\n",
			ret);
		return ret;
	}

	cdc_info.active = 1;

	return 0;
}

static const struct i2c_device_id cdc_tcxo_id[] = {
	{"cdc_tcxo_driver", 0},
	{ }
};

static struct i2c_driver cdc_tcxo_i2c_driver = {
	.driver = {
		.name		= DRIVER_NAME,
	 },
	.probe		= cdc_tcxo_probe,
	.id_table	= cdc_tcxo_id,
};

static int __init cdc_tcxo_init(void)
{
	int r;

	r = i2c_add_driver(&cdc_tcxo_i2c_driver);
	if (r < 0) {
		printk(KERN_WARNING DRIVER_NAME
		       " driver registration failed\n");
		return r;
	}

	return 0;
}

static void __exit cdc_tcxo_exit(void)
{
	i2c_del_driver(&cdc_tcxo_i2c_driver);
}


module_init(cdc_tcxo_init);
module_exit(cdc_tcxo_exit);