| // SPDX-License-Identifier: GPL-2.0-only |
| /* |
| * linux/drivers/video/omap2/omapfb-sysfs.c |
| * |
| * Copyright (C) 2008 Nokia Corporation |
| * Author: Tomi Valkeinen <tomi.valkeinen@nokia.com> |
| * |
| * Some code and ideas taken from drivers/video/omap/ driver |
| * by Imre Deak. |
| */ |
| |
| #include <linux/fb.h> |
| #include <linux/sysfs.h> |
| #include <linux/device.h> |
| #include <linux/uaccess.h> |
| #include <linux/platform_device.h> |
| #include <linux/kernel.h> |
| #include <linux/mm.h> |
| #include <linux/omapfb.h> |
| |
| #include <video/omapfb_dss.h> |
| #include <video/omapvrfb.h> |
| |
| #include "omapfb.h" |
| |
| static ssize_t show_rotate_type(struct device *dev, |
| struct device_attribute *attr, char *buf) |
| { |
| struct fb_info *fbi = dev_get_drvdata(dev); |
| struct omapfb_info *ofbi = FB2OFB(fbi); |
| |
| return snprintf(buf, PAGE_SIZE, "%d\n", ofbi->rotation_type); |
| } |
| |
| static ssize_t store_rotate_type(struct device *dev, |
| struct device_attribute *attr, |
| const char *buf, size_t count) |
| { |
| struct fb_info *fbi = dev_get_drvdata(dev); |
| struct omapfb_info *ofbi = FB2OFB(fbi); |
| struct omapfb2_mem_region *rg; |
| int rot_type; |
| int r; |
| |
| r = kstrtoint(buf, 0, &rot_type); |
| if (r) |
| return r; |
| |
| if (rot_type != OMAP_DSS_ROT_DMA && rot_type != OMAP_DSS_ROT_VRFB) |
| return -EINVAL; |
| |
| if (!lock_fb_info(fbi)) |
| return -ENODEV; |
| |
| r = 0; |
| if (rot_type == ofbi->rotation_type) |
| goto out; |
| |
| rg = omapfb_get_mem_region(ofbi->region); |
| |
| if (rg->size) { |
| r = -EBUSY; |
| goto put_region; |
| } |
| |
| ofbi->rotation_type = rot_type; |
| |
| /* |
| * Since the VRAM for this FB is not allocated at the moment we don't |
| * need to do any further parameter checking at this point. |
| */ |
| put_region: |
| omapfb_put_mem_region(rg); |
| out: |
| unlock_fb_info(fbi); |
| |
| return r ? r : count; |
| } |
| |
| |
| static ssize_t show_mirror(struct device *dev, |
| struct device_attribute *attr, char *buf) |
| { |
| struct fb_info *fbi = dev_get_drvdata(dev); |
| struct omapfb_info *ofbi = FB2OFB(fbi); |
| |
| return snprintf(buf, PAGE_SIZE, "%d\n", ofbi->mirror); |
| } |
| |
| static ssize_t store_mirror(struct device *dev, |
| struct device_attribute *attr, |
| const char *buf, size_t count) |
| { |
| struct fb_info *fbi = dev_get_drvdata(dev); |
| struct omapfb_info *ofbi = FB2OFB(fbi); |
| bool mirror; |
| int r; |
| struct fb_var_screeninfo new_var; |
| |
| r = strtobool(buf, &mirror); |
| if (r) |
| return r; |
| |
| if (!lock_fb_info(fbi)) |
| return -ENODEV; |
| |
| ofbi->mirror = mirror; |
| |
| omapfb_get_mem_region(ofbi->region); |
| |
| memcpy(&new_var, &fbi->var, sizeof(new_var)); |
| r = check_fb_var(fbi, &new_var); |
| if (r) |
| goto out; |
| memcpy(&fbi->var, &new_var, sizeof(fbi->var)); |
| |
| set_fb_fix(fbi); |
| |
| r = omapfb_apply_changes(fbi, 0); |
| if (r) |
| goto out; |
| |
| r = count; |
| out: |
| omapfb_put_mem_region(ofbi->region); |
| |
| unlock_fb_info(fbi); |
| |
| return r; |
| } |
| |
| static ssize_t show_overlays(struct device *dev, |
| struct device_attribute *attr, char *buf) |
| { |
| struct fb_info *fbi = dev_get_drvdata(dev); |
| struct omapfb_info *ofbi = FB2OFB(fbi); |
| struct omapfb2_device *fbdev = ofbi->fbdev; |
| ssize_t l = 0; |
| int t; |
| |
| if (!lock_fb_info(fbi)) |
| return -ENODEV; |
| omapfb_lock(fbdev); |
| |
| for (t = 0; t < ofbi->num_overlays; t++) { |
| struct omap_overlay *ovl = ofbi->overlays[t]; |
| int ovlnum; |
| |
| for (ovlnum = 0; ovlnum < fbdev->num_overlays; ++ovlnum) |
| if (ovl == fbdev->overlays[ovlnum]) |
| break; |
| |
| l += snprintf(buf + l, PAGE_SIZE - l, "%s%d", |
| t == 0 ? "" : ",", ovlnum); |
| } |
| |
| l += snprintf(buf + l, PAGE_SIZE - l, "\n"); |
| |
| omapfb_unlock(fbdev); |
| unlock_fb_info(fbi); |
| |
| return l; |
| } |
| |
| static struct omapfb_info *get_overlay_fb(struct omapfb2_device *fbdev, |
| struct omap_overlay *ovl) |
| { |
| int i, t; |
| |
| for (i = 0; i < fbdev->num_fbs; i++) { |
| struct omapfb_info *ofbi = FB2OFB(fbdev->fbs[i]); |
| |
| for (t = 0; t < ofbi->num_overlays; t++) { |
| if (ofbi->overlays[t] == ovl) |
| return ofbi; |
| } |
| } |
| |
| return NULL; |
| } |
| |
| static ssize_t store_overlays(struct device *dev, struct device_attribute *attr, |
| const char *buf, size_t count) |
| { |
| struct fb_info *fbi = dev_get_drvdata(dev); |
| struct omapfb_info *ofbi = FB2OFB(fbi); |
| struct omapfb2_device *fbdev = ofbi->fbdev; |
| struct omap_overlay *ovls[OMAPFB_MAX_OVL_PER_FB]; |
| struct omap_overlay *ovl; |
| int num_ovls, r, i; |
| int len; |
| bool added = false; |
| |
| num_ovls = 0; |
| |
| len = strlen(buf); |
| if (buf[len - 1] == '\n') |
| len = len - 1; |
| |
| if (!lock_fb_info(fbi)) |
| return -ENODEV; |
| omapfb_lock(fbdev); |
| |
| if (len > 0) { |
| char *p = (char *)buf; |
| int ovlnum; |
| |
| while (p < buf + len) { |
| int found; |
| if (num_ovls == OMAPFB_MAX_OVL_PER_FB) { |
| r = -EINVAL; |
| goto out; |
| } |
| |
| ovlnum = simple_strtoul(p, &p, 0); |
| if (ovlnum > fbdev->num_overlays) { |
| r = -EINVAL; |
| goto out; |
| } |
| |
| found = 0; |
| for (i = 0; i < num_ovls; ++i) { |
| if (ovls[i] == fbdev->overlays[ovlnum]) { |
| found = 1; |
| break; |
| } |
| } |
| |
| if (!found) |
| ovls[num_ovls++] = fbdev->overlays[ovlnum]; |
| |
| p++; |
| } |
| } |
| |
| for (i = 0; i < num_ovls; ++i) { |
| struct omapfb_info *ofbi2 = get_overlay_fb(fbdev, ovls[i]); |
| if (ofbi2 && ofbi2 != ofbi) { |
| dev_err(fbdev->dev, "overlay already in use\n"); |
| r = -EINVAL; |
| goto out; |
| } |
| } |
| |
| /* detach unused overlays */ |
| for (i = 0; i < ofbi->num_overlays; ++i) { |
| int t, found; |
| |
| ovl = ofbi->overlays[i]; |
| |
| found = 0; |
| |
| for (t = 0; t < num_ovls; ++t) { |
| if (ovl == ovls[t]) { |
| found = 1; |
| break; |
| } |
| } |
| |
| if (found) |
| continue; |
| |
| DBG("detaching %d\n", ofbi->overlays[i]->id); |
| |
| omapfb_get_mem_region(ofbi->region); |
| |
| omapfb_overlay_enable(ovl, 0); |
| |
| if (ovl->manager) |
| ovl->manager->apply(ovl->manager); |
| |
| omapfb_put_mem_region(ofbi->region); |
| |
| for (t = i + 1; t < ofbi->num_overlays; t++) { |
| ofbi->rotation[t-1] = ofbi->rotation[t]; |
| ofbi->overlays[t-1] = ofbi->overlays[t]; |
| } |
| |
| ofbi->num_overlays--; |
| i--; |
| } |
| |
| for (i = 0; i < num_ovls; ++i) { |
| int t, found; |
| |
| ovl = ovls[i]; |
| |
| found = 0; |
| |
| for (t = 0; t < ofbi->num_overlays; ++t) { |
| if (ovl == ofbi->overlays[t]) { |
| found = 1; |
| break; |
| } |
| } |
| |
| if (found) |
| continue; |
| ofbi->rotation[ofbi->num_overlays] = 0; |
| ofbi->overlays[ofbi->num_overlays++] = ovl; |
| |
| added = true; |
| } |
| |
| if (added) { |
| omapfb_get_mem_region(ofbi->region); |
| |
| r = omapfb_apply_changes(fbi, 0); |
| |
| omapfb_put_mem_region(ofbi->region); |
| |
| if (r) |
| goto out; |
| } |
| |
| r = count; |
| out: |
| omapfb_unlock(fbdev); |
| unlock_fb_info(fbi); |
| |
| return r; |
| } |
| |
| static ssize_t show_overlays_rotate(struct device *dev, |
| struct device_attribute *attr, char *buf) |
| { |
| struct fb_info *fbi = dev_get_drvdata(dev); |
| struct omapfb_info *ofbi = FB2OFB(fbi); |
| ssize_t l = 0; |
| int t; |
| |
| if (!lock_fb_info(fbi)) |
| return -ENODEV; |
| |
| for (t = 0; t < ofbi->num_overlays; t++) { |
| l += snprintf(buf + l, PAGE_SIZE - l, "%s%d", |
| t == 0 ? "" : ",", ofbi->rotation[t]); |
| } |
| |
| l += snprintf(buf + l, PAGE_SIZE - l, "\n"); |
| |
| unlock_fb_info(fbi); |
| |
| return l; |
| } |
| |
| static ssize_t store_overlays_rotate(struct device *dev, |
| struct device_attribute *attr, const char *buf, size_t count) |
| { |
| struct fb_info *fbi = dev_get_drvdata(dev); |
| struct omapfb_info *ofbi = FB2OFB(fbi); |
| int num_ovls = 0, r, i; |
| int len; |
| bool changed = false; |
| u8 rotation[OMAPFB_MAX_OVL_PER_FB]; |
| |
| len = strlen(buf); |
| if (buf[len - 1] == '\n') |
| len = len - 1; |
| |
| if (!lock_fb_info(fbi)) |
| return -ENODEV; |
| |
| if (len > 0) { |
| char *p = (char *)buf; |
| |
| while (p < buf + len) { |
| int rot; |
| |
| if (num_ovls == ofbi->num_overlays) { |
| r = -EINVAL; |
| goto out; |
| } |
| |
| rot = simple_strtoul(p, &p, 0); |
| if (rot < 0 || rot > 3) { |
| r = -EINVAL; |
| goto out; |
| } |
| |
| if (ofbi->rotation[num_ovls] != rot) |
| changed = true; |
| |
| rotation[num_ovls++] = rot; |
| |
| p++; |
| } |
| } |
| |
| if (num_ovls != ofbi->num_overlays) { |
| r = -EINVAL; |
| goto out; |
| } |
| |
| if (changed) { |
| for (i = 0; i < num_ovls; ++i) |
| ofbi->rotation[i] = rotation[i]; |
| |
| omapfb_get_mem_region(ofbi->region); |
| |
| r = omapfb_apply_changes(fbi, 0); |
| |
| omapfb_put_mem_region(ofbi->region); |
| |
| if (r) |
| goto out; |
| |
| /* FIXME error handling? */ |
| } |
| |
| r = count; |
| out: |
| unlock_fb_info(fbi); |
| |
| return r; |
| } |
| |
| static ssize_t show_size(struct device *dev, |
| struct device_attribute *attr, char *buf) |
| { |
| struct fb_info *fbi = dev_get_drvdata(dev); |
| struct omapfb_info *ofbi = FB2OFB(fbi); |
| |
| return snprintf(buf, PAGE_SIZE, "%lu\n", ofbi->region->size); |
| } |
| |
| static ssize_t store_size(struct device *dev, struct device_attribute *attr, |
| const char *buf, size_t count) |
| { |
| struct fb_info *fbi = dev_get_drvdata(dev); |
| struct omapfb_info *ofbi = FB2OFB(fbi); |
| struct omapfb2_device *fbdev = ofbi->fbdev; |
| struct omap_dss_device *display = fb2display(fbi); |
| struct omapfb2_mem_region *rg; |
| unsigned long size; |
| int r; |
| int i; |
| |
| r = kstrtoul(buf, 0, &size); |
| if (r) |
| return r; |
| |
| size = PAGE_ALIGN(size); |
| |
| if (!lock_fb_info(fbi)) |
| return -ENODEV; |
| |
| if (display && display->driver->sync) |
| display->driver->sync(display); |
| |
| rg = ofbi->region; |
| |
| down_write_nested(&rg->lock, rg->id); |
| atomic_inc(&rg->lock_count); |
| |
| if (atomic_read(&rg->map_count)) { |
| r = -EBUSY; |
| goto out; |
| } |
| |
| for (i = 0; i < fbdev->num_fbs; i++) { |
| struct omapfb_info *ofbi2 = FB2OFB(fbdev->fbs[i]); |
| int j; |
| |
| if (ofbi2->region != rg) |
| continue; |
| |
| for (j = 0; j < ofbi2->num_overlays; j++) { |
| struct omap_overlay *ovl; |
| ovl = ofbi2->overlays[j]; |
| if (ovl->is_enabled(ovl)) { |
| r = -EBUSY; |
| goto out; |
| } |
| } |
| } |
| |
| if (size != ofbi->region->size) { |
| r = omapfb_realloc_fbmem(fbi, size, ofbi->region->type); |
| if (r) { |
| dev_err(dev, "realloc fbmem failed\n"); |
| goto out; |
| } |
| } |
| |
| r = count; |
| out: |
| atomic_dec(&rg->lock_count); |
| up_write(&rg->lock); |
| |
| unlock_fb_info(fbi); |
| |
| return r; |
| } |
| |
| static ssize_t show_phys(struct device *dev, |
| struct device_attribute *attr, char *buf) |
| { |
| struct fb_info *fbi = dev_get_drvdata(dev); |
| struct omapfb_info *ofbi = FB2OFB(fbi); |
| |
| return snprintf(buf, PAGE_SIZE, "%0x\n", ofbi->region->paddr); |
| } |
| |
| static ssize_t show_virt(struct device *dev, |
| struct device_attribute *attr, char *buf) |
| { |
| struct fb_info *fbi = dev_get_drvdata(dev); |
| struct omapfb_info *ofbi = FB2OFB(fbi); |
| |
| return snprintf(buf, PAGE_SIZE, "%p\n", ofbi->region->vaddr); |
| } |
| |
| static ssize_t show_upd_mode(struct device *dev, |
| struct device_attribute *attr, char *buf) |
| { |
| struct fb_info *fbi = dev_get_drvdata(dev); |
| enum omapfb_update_mode mode; |
| int r; |
| |
| r = omapfb_get_update_mode(fbi, &mode); |
| |
| if (r) |
| return r; |
| |
| return snprintf(buf, PAGE_SIZE, "%u\n", (unsigned)mode); |
| } |
| |
| static ssize_t store_upd_mode(struct device *dev, struct device_attribute *attr, |
| const char *buf, size_t count) |
| { |
| struct fb_info *fbi = dev_get_drvdata(dev); |
| unsigned mode; |
| int r; |
| |
| r = kstrtouint(buf, 0, &mode); |
| if (r) |
| return r; |
| |
| r = omapfb_set_update_mode(fbi, mode); |
| if (r) |
| return r; |
| |
| return count; |
| } |
| |
| static struct device_attribute omapfb_attrs[] = { |
| __ATTR(rotate_type, S_IRUGO | S_IWUSR, show_rotate_type, |
| store_rotate_type), |
| __ATTR(mirror, S_IRUGO | S_IWUSR, show_mirror, store_mirror), |
| __ATTR(size, S_IRUGO | S_IWUSR, show_size, store_size), |
| __ATTR(overlays, S_IRUGO | S_IWUSR, show_overlays, store_overlays), |
| __ATTR(overlays_rotate, S_IRUGO | S_IWUSR, show_overlays_rotate, |
| store_overlays_rotate), |
| __ATTR(phys_addr, S_IRUGO, show_phys, NULL), |
| __ATTR(virt_addr, S_IRUGO, show_virt, NULL), |
| __ATTR(update_mode, S_IRUGO | S_IWUSR, show_upd_mode, store_upd_mode), |
| }; |
| |
| int omapfb_create_sysfs(struct omapfb2_device *fbdev) |
| { |
| int i; |
| int r; |
| |
| DBG("create sysfs for fbs\n"); |
| for (i = 0; i < fbdev->num_fbs; i++) { |
| int t; |
| for (t = 0; t < ARRAY_SIZE(omapfb_attrs); t++) { |
| r = device_create_file(fbdev->fbs[i]->dev, |
| &omapfb_attrs[t]); |
| |
| if (r) { |
| dev_err(fbdev->dev, "failed to create sysfs " |
| "file\n"); |
| return r; |
| } |
| } |
| } |
| |
| return 0; |
| } |
| |
| void omapfb_remove_sysfs(struct omapfb2_device *fbdev) |
| { |
| int i, t; |
| |
| DBG("remove sysfs for fbs\n"); |
| for (i = 0; i < fbdev->num_fbs; i++) { |
| for (t = 0; t < ARRAY_SIZE(omapfb_attrs); t++) |
| device_remove_file(fbdev->fbs[i]->dev, |
| &omapfb_attrs[t]); |
| } |
| } |
| |