tegrakernel/kernel/nvidia/drivers/video/backlight/ds1050_bl.c

163 lines
3.6 KiB
C

/*
* Maxim DS1050 Pulse-Width Modulator/Backlight Driver
*
* Copyright (C) 2015-2017 NVIDIA Corporation. All rights reserved.
*
* This program is free software; you can redistribute it and/or modify it
* under the terms and conditions of the GNU General Public License,
* version 2, as published by the Free Software Foundation.
*
* This program is distributed in the hope it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
* more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#include <linux/module.h>
#include <linux/slab.h>
#include <linux/i2c.h>
#include <linux/backlight.h>
#include <linux/err.h>
#include <linux/of.h>
#define DEFAULT_BL_NAME "pwm-backlight"
#define MAX_BRIGHTNESS 255
struct ds1050 {
struct i2c_client *client;
struct backlight_device *bl;
struct device *dev;
};
static int ds1050_write_byte(struct ds1050 *ds, u8 data)
{
return i2c_smbus_write_byte(ds->client, data);
}
static int ds1050_bl_update_status(struct backlight_device *bl)
{
struct ds1050 *ds = bl_get_data(bl);
u8 val;
if (bl->props.state & (BL_CORE_SUSPENDED | BL_CORE_FBBLANK))
bl->props.brightness = 0;
val = bl->props.brightness;
/* DS1050 is a 5-bit PWM:
* 0b00000 - 0% duty cycle
* ...
* 0b11111 - 96.88% duty cycle
* 0b100000 - 100% duty cycle
*/
if (val == MAX_BRIGHTNESS)
val = 0x20;
else
val >>= 3; /* ignore the 3 least significant bits */
ds1050_write_byte(ds, val);
return 0;
}
static const struct backlight_ops ds1050_bl_ops = {
.options = BL_CORE_SUSPENDRESUME,
.update_status = ds1050_bl_update_status,
};
static int ds1050_backlight_register(struct ds1050 *ds)
{
struct backlight_device *bl;
struct backlight_properties props;
const char *name = DEFAULT_BL_NAME;
memset(&props, 0, sizeof(props));
props.type = BACKLIGHT_PLATFORM;
props.max_brightness = MAX_BRIGHTNESS;
props.brightness = MAX_BRIGHTNESS / 2;
bl = devm_backlight_device_register(ds->dev, name, ds->dev, ds,
&ds1050_bl_ops, &props);
if (IS_ERR(bl))
return PTR_ERR(bl);
ds->bl = bl;
backlight_update_status(ds->bl);
return 0;
}
static void ds1050_backlight_unregister(struct ds1050 *ds)
{
if (!ds)
return;
ds->bl->props.brightness = 0;
backlight_update_status(ds->bl);
devm_backlight_device_unregister(ds->dev, ds->bl);
}
static int ds1050_probe(struct i2c_client *cl, const struct i2c_device_id *id)
{
struct ds1050 *ds;
int ret = 0;
ds = devm_kzalloc(&cl->dev, sizeof(struct ds1050), GFP_KERNEL);
if (!ds)
return -ENOMEM;
ds->client = cl;
ds->dev = &cl->dev;
i2c_set_clientdata(cl, ds);
ret = ds1050_backlight_register(ds);
if (ret)
dev_err(ds->dev,
"failed to register backlight, err: %d\n", ret);
return ret;
}
static int ds1050_remove(struct i2c_client *cl)
{
struct ds1050 *ds = i2c_get_clientdata(cl);
ds1050_backlight_unregister(ds);
devm_kfree(&cl->dev, ds);
return 0;
}
static const struct of_device_id ds1050_dt_ids[] = {
{ .compatible = "maxim,ds1050", },
{ }
};
MODULE_DEVICE_TABLE(of, ds1050_dt_ids);
static const struct i2c_device_id ds1050_ids[] = {
{"ds1050", 0},
{ }
};
MODULE_DEVICE_TABLE(i2c, ds1050_ids);
static struct i2c_driver ds1050_driver = {
.driver = {
.name = "ds1050",
.of_match_table = of_match_ptr(ds1050_dt_ids),
},
.probe = ds1050_probe,
.remove = ds1050_remove,
.id_table = ds1050_ids,
};
module_i2c_driver(ds1050_driver);
MODULE_DESCRIPTION("Maxim DS1050 Backlight driver");
MODULE_LICENSE("GPL");