xref: /openbmc/linux/drivers/hid/hid-picolcd_lcd.c (revision d881427253da011495f4193663d809d0e9dfa215)
1fabdbf2fSBruno Prémont /***************************************************************************
2fabdbf2fSBruno Prémont  *   Copyright (C) 2010-2012 by Bruno Prémont <bonbons@linux-vserver.org>  *
3fabdbf2fSBruno Prémont  *                                                                         *
4fabdbf2fSBruno Prémont  *   Based on Logitech G13 driver (v0.4)                                   *
5fabdbf2fSBruno Prémont  *     Copyright (C) 2009 by Rick L. Vinyard, Jr. <rvinyard@cs.nmsu.edu>   *
6fabdbf2fSBruno Prémont  *                                                                         *
7fabdbf2fSBruno Prémont  *   This program is free software: you can redistribute it and/or modify  *
8fabdbf2fSBruno Prémont  *   it under the terms of the GNU General Public License as published by  *
9fabdbf2fSBruno Prémont  *   the Free Software Foundation, version 2 of the License.               *
10fabdbf2fSBruno Prémont  *                                                                         *
11fabdbf2fSBruno Prémont  *   This driver is distributed in the hope that it will be useful, but    *
12fabdbf2fSBruno Prémont  *   WITHOUT ANY WARRANTY; without even the implied warranty of            *
13fabdbf2fSBruno Prémont  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU      *
14fabdbf2fSBruno Prémont  *   General Public License for more details.                              *
15fabdbf2fSBruno Prémont  *                                                                         *
16fabdbf2fSBruno Prémont  *   You should have received a copy of the GNU General Public License     *
17fabdbf2fSBruno Prémont  *   along with this software. If not see <http://www.gnu.org/licenses/>.  *
18fabdbf2fSBruno Prémont  ***************************************************************************/
19fabdbf2fSBruno Prémont 
20fabdbf2fSBruno Prémont #include <linux/hid.h>
21fabdbf2fSBruno Prémont 
22fabdbf2fSBruno Prémont #include <linux/fb.h>
23fabdbf2fSBruno Prémont #include <linux/lcd.h>
24fabdbf2fSBruno Prémont 
25fabdbf2fSBruno Prémont #include "hid-picolcd.h"
26fabdbf2fSBruno Prémont 
27fabdbf2fSBruno Prémont /*
28fabdbf2fSBruno Prémont  * lcd class device
29fabdbf2fSBruno Prémont  */
30fabdbf2fSBruno Prémont static int picolcd_get_contrast(struct lcd_device *ldev)
31fabdbf2fSBruno Prémont {
32fabdbf2fSBruno Prémont 	struct picolcd_data *data = lcd_get_data(ldev);
33fabdbf2fSBruno Prémont 	return data->lcd_contrast;
34fabdbf2fSBruno Prémont }
35fabdbf2fSBruno Prémont 
36fabdbf2fSBruno Prémont static int picolcd_set_contrast(struct lcd_device *ldev, int contrast)
37fabdbf2fSBruno Prémont {
38fabdbf2fSBruno Prémont 	struct picolcd_data *data = lcd_get_data(ldev);
39fabdbf2fSBruno Prémont 	struct hid_report *report = picolcd_out_report(REPORT_CONTRAST, data->hdev);
40fabdbf2fSBruno Prémont 	unsigned long flags;
41fabdbf2fSBruno Prémont 
42fabdbf2fSBruno Prémont 	if (!report || report->maxfield != 1 || report->field[0]->report_count != 1)
43fabdbf2fSBruno Prémont 		return -ENODEV;
44fabdbf2fSBruno Prémont 
45fabdbf2fSBruno Prémont 	data->lcd_contrast = contrast & 0x0ff;
46fabdbf2fSBruno Prémont 	spin_lock_irqsave(&data->lock, flags);
47fabdbf2fSBruno Prémont 	hid_set_field(report->field[0], 0, data->lcd_contrast);
48a93ab849SBruno Prémont 	if (!(data->status & PICOLCD_FAILED))
49*d8814272SBenjamin Tissoires 		hid_hw_request(data->hdev, report, HID_REQ_SET_REPORT);
50fabdbf2fSBruno Prémont 	spin_unlock_irqrestore(&data->lock, flags);
51fabdbf2fSBruno Prémont 	return 0;
52fabdbf2fSBruno Prémont }
53fabdbf2fSBruno Prémont 
54fabdbf2fSBruno Prémont static int picolcd_check_lcd_fb(struct lcd_device *ldev, struct fb_info *fb)
55fabdbf2fSBruno Prémont {
56fabdbf2fSBruno Prémont 	return fb && fb == picolcd_fbinfo((struct picolcd_data *)lcd_get_data(ldev));
57fabdbf2fSBruno Prémont }
58fabdbf2fSBruno Prémont 
59fabdbf2fSBruno Prémont static struct lcd_ops picolcd_lcdops = {
60fabdbf2fSBruno Prémont 	.get_contrast   = picolcd_get_contrast,
61fabdbf2fSBruno Prémont 	.set_contrast   = picolcd_set_contrast,
62fabdbf2fSBruno Prémont 	.check_fb       = picolcd_check_lcd_fb,
63fabdbf2fSBruno Prémont };
64fabdbf2fSBruno Prémont 
65fabdbf2fSBruno Prémont int picolcd_init_lcd(struct picolcd_data *data, struct hid_report *report)
66fabdbf2fSBruno Prémont {
67fabdbf2fSBruno Prémont 	struct device *dev = &data->hdev->dev;
68fabdbf2fSBruno Prémont 	struct lcd_device *ldev;
69fabdbf2fSBruno Prémont 
70fabdbf2fSBruno Prémont 	if (!report)
71fabdbf2fSBruno Prémont 		return -ENODEV;
72fabdbf2fSBruno Prémont 	if (report->maxfield != 1 || report->field[0]->report_count != 1 ||
73fabdbf2fSBruno Prémont 			report->field[0]->report_size != 8) {
74fabdbf2fSBruno Prémont 		dev_err(dev, "unsupported CONTRAST report");
75fabdbf2fSBruno Prémont 		return -EINVAL;
76fabdbf2fSBruno Prémont 	}
77fabdbf2fSBruno Prémont 
78fabdbf2fSBruno Prémont 	ldev = lcd_device_register(dev_name(dev), dev, data, &picolcd_lcdops);
79fabdbf2fSBruno Prémont 	if (IS_ERR(ldev)) {
80fabdbf2fSBruno Prémont 		dev_err(dev, "failed to register LCD\n");
81fabdbf2fSBruno Prémont 		return PTR_ERR(ldev);
82fabdbf2fSBruno Prémont 	}
83fabdbf2fSBruno Prémont 	ldev->props.max_contrast = 0x0ff;
84fabdbf2fSBruno Prémont 	data->lcd_contrast = 0xe5;
85fabdbf2fSBruno Prémont 	data->lcd = ldev;
86fabdbf2fSBruno Prémont 	picolcd_set_contrast(ldev, 0xe5);
87fabdbf2fSBruno Prémont 	return 0;
88fabdbf2fSBruno Prémont }
89fabdbf2fSBruno Prémont 
90fabdbf2fSBruno Prémont void picolcd_exit_lcd(struct picolcd_data *data)
91fabdbf2fSBruno Prémont {
92fabdbf2fSBruno Prémont 	struct lcd_device *ldev = data->lcd;
93fabdbf2fSBruno Prémont 
94fabdbf2fSBruno Prémont 	data->lcd = NULL;
95fabdbf2fSBruno Prémont 	if (ldev)
96fabdbf2fSBruno Prémont 		lcd_device_unregister(ldev);
97fabdbf2fSBruno Prémont }
98fabdbf2fSBruno Prémont 
99fabdbf2fSBruno Prémont int picolcd_resume_lcd(struct picolcd_data *data)
100fabdbf2fSBruno Prémont {
101fabdbf2fSBruno Prémont 	if (!data->lcd)
102fabdbf2fSBruno Prémont 		return 0;
103fabdbf2fSBruno Prémont 	return picolcd_set_contrast(data->lcd, data->lcd_contrast);
104fabdbf2fSBruno Prémont }
105fabdbf2fSBruno Prémont 
106