588 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			588 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| // 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;
 | |
| 
 | |
| 	lock_fb_info(fbi);
 | |
| 
 | |
| 	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;
 | |
| 
 | |
| 	lock_fb_info(fbi);
 | |
| 
 | |
| 	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;
 | |
| 
 | |
| 	lock_fb_info(fbi);
 | |
| 	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 += scnprintf(buf + l, PAGE_SIZE - l, "%s%d",
 | |
| 				t == 0 ? "" : ",", ovlnum);
 | |
| 	}
 | |
| 
 | |
| 	l += scnprintf(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;
 | |
| 
 | |
| 	lock_fb_info(fbi);
 | |
| 	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;
 | |
| 
 | |
| 	lock_fb_info(fbi);
 | |
| 
 | |
| 	for (t = 0; t < ofbi->num_overlays; t++) {
 | |
| 		l += scnprintf(buf + l, PAGE_SIZE - l, "%s%d",
 | |
| 				t == 0 ? "" : ",", ofbi->rotation[t]);
 | |
| 	}
 | |
| 
 | |
| 	l += scnprintf(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;
 | |
| 
 | |
| 	lock_fb_info(fbi);
 | |
| 
 | |
| 	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);
 | |
| 
 | |
| 	lock_fb_info(fbi);
 | |
| 
 | |
| 	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]);
 | |
| 	}
 | |
| }
 | |
| 
 |