| 
									
										
										
										
											2013-07-04 21:13:25 +02:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Sanyo LV5207LP LED Driver | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Copyright (C) 2013 Ideas on board SPRL | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Contact: Laurent Pinchart <laurent.pinchart@ideasonboard.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/backlight.h>
 | 
					
						
							|  |  |  | #include <linux/err.h>
 | 
					
						
							|  |  |  | #include <linux/fb.h>
 | 
					
						
							|  |  |  | #include <linux/i2c.h>
 | 
					
						
							|  |  |  | #include <linux/module.h>
 | 
					
						
							|  |  |  | #include <linux/platform_data/lv5207lp.h>
 | 
					
						
							|  |  |  | #include <linux/slab.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define LV5207LP_CTRL1			0x00
 | 
					
						
							|  |  |  | #define LV5207LP_CPSW			(1 << 7)
 | 
					
						
							|  |  |  | #define LV5207LP_SCTEN			(1 << 6)
 | 
					
						
							|  |  |  | #define LV5207LP_C10			(1 << 5)
 | 
					
						
							|  |  |  | #define LV5207LP_CKSW			(1 << 4)
 | 
					
						
							|  |  |  | #define LV5207LP_RSW			(1 << 3)
 | 
					
						
							|  |  |  | #define LV5207LP_GSW			(1 << 2)
 | 
					
						
							|  |  |  | #define LV5207LP_BSW			(1 << 1)
 | 
					
						
							|  |  |  | #define LV5207LP_CTRL2			0x01
 | 
					
						
							|  |  |  | #define LV5207LP_MSW			(1 << 7)
 | 
					
						
							|  |  |  | #define LV5207LP_MLED4			(1 << 6)
 | 
					
						
							|  |  |  | #define LV5207LP_RED			0x02
 | 
					
						
							|  |  |  | #define LV5207LP_GREEN			0x03
 | 
					
						
							|  |  |  | #define LV5207LP_BLUE			0x04
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define LV5207LP_MAX_BRIGHTNESS		32
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | struct lv5207lp { | 
					
						
							|  |  |  | 	struct i2c_client *client; | 
					
						
							|  |  |  | 	struct backlight_device *backlight; | 
					
						
							|  |  |  | 	struct lv5207lp_platform_data *pdata; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int lv5207lp_write(struct lv5207lp *lv, u8 reg, u8 data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	return i2c_smbus_write_byte_data(lv->client, reg, data); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int lv5207lp_backlight_update_status(struct backlight_device *backlight) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct lv5207lp *lv = bl_get_data(backlight); | 
					
						
							|  |  |  | 	int brightness = backlight->props.brightness; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (backlight->props.power != FB_BLANK_UNBLANK || | 
					
						
							|  |  |  | 	    backlight->props.fb_blank != FB_BLANK_UNBLANK || | 
					
						
							|  |  |  | 	    backlight->props.state & (BL_CORE_SUSPENDED | BL_CORE_FBBLANK)) | 
					
						
							|  |  |  | 		brightness = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (brightness) { | 
					
						
							|  |  |  | 		lv5207lp_write(lv, LV5207LP_CTRL1, | 
					
						
							|  |  |  | 			       LV5207LP_CPSW | LV5207LP_C10 | LV5207LP_CKSW); | 
					
						
							|  |  |  | 		lv5207lp_write(lv, LV5207LP_CTRL2, | 
					
						
							|  |  |  | 			       LV5207LP_MSW | LV5207LP_MLED4 | | 
					
						
							|  |  |  | 			       (brightness - 1)); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | 		lv5207lp_write(lv, LV5207LP_CTRL1, 0); | 
					
						
							|  |  |  | 		lv5207lp_write(lv, LV5207LP_CTRL2, 0); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int lv5207lp_backlight_check_fb(struct backlight_device *backlight, | 
					
						
							|  |  |  | 				       struct fb_info *info) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct lv5207lp *lv = bl_get_data(backlight); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return lv->pdata->fbdev == NULL || lv->pdata->fbdev == info->dev; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const struct backlight_ops lv5207lp_backlight_ops = { | 
					
						
							|  |  |  | 	.options	= BL_CORE_SUSPENDRESUME, | 
					
						
							|  |  |  | 	.update_status	= lv5207lp_backlight_update_status, | 
					
						
							|  |  |  | 	.check_fb	= lv5207lp_backlight_check_fb, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int lv5207lp_probe(struct i2c_client *client, | 
					
						
							|  |  |  | 			  const struct i2c_device_id *id) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2013-11-12 15:09:04 -08:00
										 |  |  | 	struct lv5207lp_platform_data *pdata = dev_get_platdata(&client->dev); | 
					
						
							| 
									
										
										
										
											2013-07-04 21:13:25 +02:00
										 |  |  | 	struct backlight_device *backlight; | 
					
						
							|  |  |  | 	struct backlight_properties props; | 
					
						
							|  |  |  | 	struct lv5207lp *lv; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (pdata == NULL) { | 
					
						
							|  |  |  | 		dev_err(&client->dev, "No platform data supplied\n"); | 
					
						
							|  |  |  | 		return -EINVAL; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if (!i2c_check_functionality(client->adapter, | 
					
						
							|  |  |  | 				     I2C_FUNC_SMBUS_BYTE_DATA)) { | 
					
						
							|  |  |  | 		dev_warn(&client->dev, | 
					
						
							|  |  |  | 			 "I2C adapter doesn't support I2C_FUNC_SMBUS_BYTE\n"); | 
					
						
							|  |  |  | 		return -EIO; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	lv = devm_kzalloc(&client->dev, sizeof(*lv), GFP_KERNEL); | 
					
						
							|  |  |  | 	if (!lv) | 
					
						
							|  |  |  | 		return -ENOMEM; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	lv->client = client; | 
					
						
							|  |  |  | 	lv->pdata = pdata; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	memset(&props, 0, sizeof(props)); | 
					
						
							|  |  |  | 	props.type = BACKLIGHT_RAW; | 
					
						
							|  |  |  | 	props.max_brightness = min_t(unsigned int, pdata->max_value, | 
					
						
							|  |  |  | 				     LV5207LP_MAX_BRIGHTNESS); | 
					
						
							|  |  |  | 	props.brightness = clamp_t(unsigned int, pdata->def_value, 0, | 
					
						
							|  |  |  | 				   props.max_brightness); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-12 15:09:20 -08:00
										 |  |  | 	backlight = devm_backlight_device_register(&client->dev, | 
					
						
							|  |  |  | 				dev_name(&client->dev), &lv->client->dev, | 
					
						
							|  |  |  | 				lv, &lv5207lp_backlight_ops, &props); | 
					
						
							| 
									
										
										
										
											2013-07-04 21:13:25 +02:00
										 |  |  | 	if (IS_ERR(backlight)) { | 
					
						
							|  |  |  | 		dev_err(&client->dev, "failed to register backlight\n"); | 
					
						
							|  |  |  | 		return PTR_ERR(backlight); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	backlight_update_status(backlight); | 
					
						
							|  |  |  | 	i2c_set_clientdata(client, backlight); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int lv5207lp_remove(struct i2c_client *client) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	struct backlight_device *backlight = i2c_get_clientdata(client); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	backlight->props.brightness = 0; | 
					
						
							|  |  |  | 	backlight_update_status(backlight); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static const struct i2c_device_id lv5207lp_ids[] = { | 
					
						
							|  |  |  | 	{ "lv5207lp", 0 }, | 
					
						
							|  |  |  | 	{ } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | MODULE_DEVICE_TABLE(i2c, lv5207lp_ids); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static struct i2c_driver lv5207lp_driver = { | 
					
						
							|  |  |  | 	.driver = { | 
					
						
							|  |  |  | 		.name = "lv5207lp", | 
					
						
							|  |  |  | 	}, | 
					
						
							|  |  |  | 	.probe = lv5207lp_probe, | 
					
						
							|  |  |  | 	.remove = lv5207lp_remove, | 
					
						
							|  |  |  | 	.id_table = lv5207lp_ids, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | module_i2c_driver(lv5207lp_driver); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | MODULE_DESCRIPTION("Sanyo LV5207LP Backlight Driver"); | 
					
						
							|  |  |  | MODULE_AUTHOR("Laurent Pinchart <laurent.pinchart@ideasonboard.com>"); | 
					
						
							|  |  |  | MODULE_LICENSE("GPL"); |