summaryrefslogtreecommitdiff
path: root/arch/arm/mach-mx6/mx6_ir.c
blob: 73d18ce266b02884e94c41e47388f36b2366e4e0 (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
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
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
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
/*
 * Copyright (C) 2005-2013 Freescale Semiconductor, Inc. All Rights Reserved.
 */
/*
 * The code contained herein is licensed under the GNU General Public
 * License. You may obtain a copy of the GNU General Public License
 * Version 2 or later at the following locations:
 *
 * http://www.opensource.org/licenses/gpl-license.html
 * http://www.gnu.org/copyleft/gpl.html
 */
#include <linux/init.h>
#include <linux/interrupt.h>
#include <linux/sched.h>
#include <linux/module.h>
#include <linux/kernel.h>
#include <linux/platform_device.h>
#include <linux/slab.h>
#include <linux/err.h>
#include <linux/clk.h>
#include <linux/io.h>
#include <linux/pwm.h>
#include <linux/fsl_devices.h>
#include <linux/ir.h>
#include <mach/hardware.h>
#include <mach/mxc_ir.h>

#ifdef IR_HRTIMER_USED
#include <linux/hrtimer.h>
#include <linux/ktime.h>
#else
#include <mach/epit.h>
#endif

#define IR_MAX_CARRIER_FREQ (11000000)
#define IR_MIN_CARRIER_FREQ (3000)

struct mx6_ir_device {
	struct pwm_device	*pwm;
#ifdef IR_HRTIMER_USED
	struct hrtimer		hr_timer;
#else
	struct epit_device	*epit;
#endif
	int					pattern_len;
	int					*pattern;
	int					transmit_idx;
	int					transmit_level;
	int					gpio_pin;
	struct device		*dev;
	wait_queue_head_t	queue;
	bool				cond;
	unsigned int		carry_freq;
};

struct carrier_freq {
	int min;
	int max;
};

static const struct carrier_freq mx6_ir_carrier_freq[] = {
	{IR_MIN_CARRIER_FREQ, IR_MAX_CARRIER_FREQ},
};

int ir_config(void *dev, int carry_freq)
{
    unsigned int period_ns;
	struct mx6_ir_device *ir_dev = (struct mx6_ir_device *)dev;

	if ((!ir_dev) || (!carry_freq))
		return -EINVAL;

	if (!ir_dev->pwm)
		return -EINVAL;

	period_ns = 1000000000;
	do_div(period_ns, carry_freq);
	pwm_config(ir_dev->pwm, (period_ns >> 1), period_ns);
	pwm_out_enable(ir_dev->pwm, 0);
	return 0;
}
EXPORT_SYMBOL(ir_config);

int ir_get_carrier_range(int id, int *min, int *max)
{
	if ((!min) || (!max)) {
		return -1;
	} else {
		*min = mx6_ir_carrier_freq[id].min;
		*max = mx6_ir_carrier_freq[id].max;
		return 0;
	}
}
EXPORT_SYMBOL(ir_get_carrier_range);

int ir_get_num_carrier_freqs(void)
{
	return ARRAY_SIZE(mx6_ir_carrier_freq);
}
EXPORT_SYMBOL(ir_get_num_carrier_freqs);


#ifdef IR_HRTIMER_USED
static void *hrtimer_cb_para;

static inline void ir_hrtimer_start(struct hrtimer *hr_timer, int width_ns,
									void *cb, void *para)
{
	ktime_t ktime;

	ktime = ktime_set(0, width_ns);
	hrtimer_init(hr_timer, CLOCK_MONOTONIC, HRTIMER_MODE_REL);
	hr_timer->function = cb;
	hrtimer_cb_para = para;
	hrtimer_start(hr_timer, ktime, HRTIMER_MODE_REL);
}

static inline void ir_hrtimer_stop(struct hrtimer *hr_timer)
{
	hrtimer_cancel(hr_timer);
}

static int ir_transmit_cb(struct hrtimer *timer)
{
	struct mx6_ir_device *ir_dev = (struct mx6_ir_device *)hrtimer_cb_para;
	ktime_t ktime;

	ir_dev->transmit_idx++;
	if (ir_dev->pattern_len == ir_dev->transmit_idx) {
		/* all bit transmit is done */
		/* stop pwm output */
		pwm_out_enable(ir_dev->pwm, 0);
		ir_dev->cond = true;
		wake_up(&ir_dev->queue);
		return HRTIMER_NORESTART;
	} else {
		/* start next bit transmit */;
		ir_dev->transmit_level = !ir_dev->transmit_level;
		ktime = ktime_set(0, ir_dev->pattern[ir_dev->transmit_idx] * 1000);
		hrtimer_forward(&ir_dev->hr_timer,
						ktime_get(),
						ktime);
		pwm_out_enable(ir_dev->pwm, ir_dev->transmit_level);
		return HRTIMER_RESTART;
	}
}

int ir_transmit(void *dev, int len, int *pattern, unsigned char start_level)
{
	struct mx6_ir_device *ir_dev = (struct mx6_ir_device *)dev;
	char level = start_level;

	if ((!ir_dev) || (!pattern))
		return -EINVAL;

	if (!ir_dev->pwm)
		return -EINVAL;

	ir_dev->transmit_level = start_level;
	ir_dev->transmit_idx = 0;
	ir_dev->pattern = pattern;
	ir_dev->pattern_len = len;
	ir_dev->cond = false;
	/*
	1. Enable/disable pwm output per level, start timer with width
	2. time cb will start next bit transmit,wait all bit transmit done
	3. disable pwm output
	*/
	pwm_enable(ir_dev->pwm);
	ir_dev->cond = false;
	ir_hrtimer_start(&ir_dev->hr_timer,
					(ir_dev->pattern[0] * 1000),
					(void *)ir_transmit_cb,
					(void *)(ir_dev));
	pwm_out_enable(ir_dev->pwm, ir_dev->transmit_level);
	wait_event(ir_dev->queue, ir_dev->cond);
	ir_hrtimer_stop(&ir_dev->hr_timer);
	pwm_disable(ir_dev->pwm);
	return 0;
}
#else
static int ir_transmit_cb(void *dev)
{
	struct mx6_ir_device *ir_dev = (struct mx6_ir_device *)dev;
	ir_dev->transmit_idx++;
	if (ir_dev->pattern_len == ir_dev->transmit_idx) {
		/* all bit transmit is done */
		/* stop pwm output */
		pwm_out_enable(ir_dev->pwm, 0);
		ir_dev->cond = true;
		wake_up(&ir_dev->queue);
	} else {
		/* start next bit transmit */;
		ir_dev->transmit_level = !ir_dev->transmit_level;
		pwm_out_enable(ir_dev->pwm, ir_dev->transmit_level);
		epit_start(ir_dev->epit,
					(ir_dev->pattern[ir_dev->transmit_idx] * 1000));

	}
	return 0;
}

int ir_transmit(void *dev, int len, int *pattern, unsigned char start_level)
{
	struct mx6_ir_device *ir_dev = (struct mx6_ir_device *)dev;

	if ((!ir_dev) || (!pattern))
		return -EINVAL;

	if ((!ir_dev->pwm) || (!ir_dev->epit))
		return -EINVAL;

	ir_dev->transmit_level = start_level;
	ir_dev->transmit_idx = 0;
	ir_dev->pattern = pattern;
	ir_dev->pattern_len = len;
	ir_dev->cond = false;
	/*
	1. Enable/disable pwm output per level, start timer with width
	2. time cb will start next bit transmit,wait all bit transmit done
	3. disable pwm output
	*/
	pwm_enable(ir_dev->pwm);
	epit_config(ir_dev->epit,
				EPIT_FREE_RUN_MODE,
				(void *)ir_transmit_cb,
				(void *)(ir_dev));
	epit_start(ir_dev->epit,
		       (ir_dev->pattern[0] * 1000));
	pwm_out_enable(ir_dev->pwm, ir_dev->transmit_level);
	wait_event(ir_dev->queue, ir_dev->cond);
	epit_stop(ir_dev->epit);
	pwm_disable(ir_dev->pwm);
	return 0;
}
#endif
EXPORT_SYMBOL(ir_transmit);

static int __devinit mx6_ir_probe(struct platform_device *pdev)
{
	struct platform_ir_data *plat_data = pdev->dev.platform_data;
	struct mx6_ir_device *pb;
	int ret = 0;

	if (!plat_data) {
		dev_err(&pdev->dev, "failed to find platform data\n");
		return -EINVAL;
	}
	pb = kzalloc(sizeof(*pb), GFP_KERNEL);
	if (!pb) {
		dev_err(&pdev->dev, "no memory for state\n");
		ret = -ENOMEM;
		goto err_alloc;
	}
	pb->dev = &pdev->dev;

	pb->pwm = pwm_request(plat_data->pwm_id, "IR Carrier");
	if (IS_ERR(pb->pwm)) {
		dev_err(&pdev->dev, "unable to request PWM for IR Carrier\n");
		ret = PTR_ERR(pb->pwm);
		goto err_pwm;
	} else {
		dev_dbg(&pdev->dev, "got pwm for IR Carrier\n");
	}

#ifndef IR_HRTIMER_USED
	pb->epit = epit_request(plat_data->epit_id, "IR Signal");
	if (IS_ERR(pb->epit)) {
		dev_err(&pdev->dev, "unable to request EPIT for IR Signal\n");
		ret = PTR_ERR(pb->epit);
		goto err_epit;
	} else {
		dev_dbg(&pdev->dev, "got EPIT for IR Signal\n");
	}
#endif

	pb->gpio_pin = plat_data->gpio_id;
	/* Init queue */
	init_waitqueue_head(&pb->queue);

	ir_device_register(dev_name(&pdev->dev), &pdev->dev, pb);
	platform_set_drvdata(pdev, pb);
	return 0;

#ifndef IR_HRTIMER_USED
err_epit:
#endif
	pwm_free(pb->pwm);
err_pwm:
	kfree(pb);
err_alloc:
	return ret;
}

static int __devexit mx6_ir_remove(struct platform_device *pdev)
{
	struct mx6_ir_device *pb = platform_get_drvdata(pdev);
	ir_device_unregister();
	pwm_disable(pb->pwm);
	pwm_free(pb->pwm);
#ifndef IR_HRTIMER_USED
	epit_stop(pb->epit);
	epit_free(pb->epit);
#endif
	kfree(pb);
	return 0;
}

static struct platform_driver mx6_ir_driver = {
	.driver = {
		.name = "mx6-ir",
	},
	.probe = mx6_ir_probe,
	.remove = __devexit_p(mx6_ir_remove),
};

static int __init mx6_ir_init(void)
{
	return platform_driver_register(&mx6_ir_driver);
}
module_init(mx6_ir_init);

static void __exit mx6_ir_exit(void)
{
	platform_driver_unregister(&mx6_ir_driver);
}
module_exit(mx6_ir_exit);

MODULE_AUTHOR("Freescale Semiconductor, Inc.");
MODULE_DESCRIPTION("MX6 IR  driver");
MODULE_LICENSE("GPL");