This patch adds sysfs handles to enable userspace control over the display contrast as well as the dim mode. The handles are available as "contrast" and "dim" in the brightness group of the framebuffers sysfs domain. Signed-off-by: Thomas Niederprüm <niederp@xxxxxxxxxxxxxxxx> --- drivers/video/fbdev/ssd1307fb.c | 90 +++++++++++++++++++++++++++++++++++++++++ 1 file changed, 90 insertions(+) diff --git a/drivers/video/fbdev/ssd1307fb.c b/drivers/video/fbdev/ssd1307fb.c index 5bf570b..2a2e7c3 100644 --- a/drivers/video/fbdev/ssd1307fb.c +++ b/drivers/video/fbdev/ssd1307fb.c @@ -33,6 +33,7 @@ #define SSD1307FB_CONTRAST 0x81 #define SSD1307FB_CHARGE_PUMP 0x8d #define SSD1307FB_SEG_REMAP_ON 0xa1 +#define SSD1307FB_DISPLAY_DIM 0xac #define SSD1307FB_DISPLAY_OFF 0xae #define SSD1307FB_SET_MULTIPLEX_RATIO 0xa8 #define SSD1307FB_DISPLAY_ON 0xaf @@ -43,6 +44,9 @@ #define SSD1307FB_SET_COM_PINS_CONFIG 0xda #define SSD1307FB_SET_VCOMH 0xdb +#define MIN_CONTRAST 0 +#define MAX_CONTRAST 255 + #define REFRASHRATE 1 #define BITSPERPIXEL 1 @@ -73,6 +77,7 @@ struct ssd1307fb_par { u32 dclk_div; u32 dclk_frq; struct ssd1307fb_deviceinfo *device_info; + u32 dim; struct i2c_client *client; u32 height; struct fb_info *info; @@ -476,6 +481,87 @@ static const struct of_device_id ssd1307fb_of_match[] = { }; MODULE_DEVICE_TABLE(of, ssd1307fb_of_match); +static ssize_t show_contrast(struct device *device, + struct device_attribute *attr, char *buf) +{ + struct fb_info *info = dev_get_drvdata(device); + struct ssd1307fb_par *par = info->par; + + return snprintf(buf, PAGE_SIZE, "%d\n", par->contrast); +} + +static ssize_t store_contrast(struct device *device, + struct device_attribute *attr, + const char *buf, size_t count) +{ + struct fb_info *info = dev_get_drvdata(device); + struct ssd1307fb_par *par = info->par; + unsigned long contrastval; + int ret; + + ret = kstrtoul(buf, 0, &contrastval); + if (ret < 0) + return ret; + + par->contrast = max_t(ulong, min_t(ulong, contrastval, + (ulong)MAX_CONTRAST), (ulong)MIN_CONTRAST); + + ret = ssd1307fb_write_cmd(par->client, SSD1307FB_CONTRAST); + ret = ret & ssd1307fb_write_cmd(par->client, par->contrast); + if (ret < 0) + return ret; + + return count; +} + + +static ssize_t show_dim(struct device *device, + struct device_attribute *attr, char *buf) +{ + struct fb_info *info = dev_get_drvdata(device); + struct ssd1307fb_par *par = info->par; + + return snprintf(buf, PAGE_SIZE, "%d\n", par->dim); +} + +static ssize_t store_dim(struct device *device, + struct device_attribute *attr, + const char *buf, size_t count) +{ + struct fb_info *info = dev_get_drvdata(device); + struct ssd1307fb_par *par = info->par; + unsigned long dimval; + int ret; + + ret = kstrtoul(buf, 0, &dimval); + if (ret < 0) + return ret; + + par->dim = max_t(ulong, min_t(ulong, dimval, (ulong)1), (ulong)0); + if (par->dim) + ret = ssd1307fb_write_cmd(par->client, SSD1307FB_DISPLAY_DIM); + else + ret = ssd1307fb_write_cmd(par->client, SSD1307FB_DISPLAY_ON); + if (ret < 0) + return ret; + + return count; +} + +static DEVICE_ATTR(contrast, S_IRUGO|S_IWUSR, show_contrast, store_contrast); +static DEVICE_ATTR(dim, S_IRUGO|S_IWUSR, show_dim, store_dim); + +static struct attribute *panel_attrs[] = { + &dev_attr_contrast.attr, + &dev_attr_dim.attr, + NULL, +}; + +static struct attribute_group brightness_attr_grp = { + .name = "brightness", + .attrs = panel_attrs, +}; + static int ssd1307fb_probe(struct i2c_client *client, const struct i2c_device_id *id) { @@ -614,6 +700,10 @@ static int ssd1307fb_probe(struct i2c_client *client, goto panel_init_error; } + ret = sysfs_create_group(&info->dev->kobj, &brightness_attr_grp); + if (ret) + dev_err(&client->dev, "Couldn't register sysfs nodes\n"); + dev_info(&client->dev, "fb%d: %s framebuffer device registered, using %d bytes of video memory\n", info->node, info->fix.id, vmem_size); return 0; -- 2.3.0 -- To unsubscribe from this list: send the line "unsubscribe linux-fbdev" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html