Release 4.9 drivers/video/fbdev/omap2/omapfb/omapfb-sysfs.c
  
  
/*
 * 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.
 *
 * 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.
 *
 * This program is distributed in the hope that 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/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);
}
Contributors
 | Person | Tokens | Prop | Commits | CommitProp | 
| tomi valkeinen | tomi valkeinen | 54 | 100.00% | 1 | 100.00% | 
 | Total | 54 | 100.00% | 1 | 100.00% | 
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;
}
Contributors
 | Person | Tokens | Prop | Commits | CommitProp | 
| tomi valkeinen | tomi valkeinen | 142 | 84.52% | 3 | 50.00% | 
| ville syrjala | ville syrjala | 18 | 10.71% | 2 | 33.33% | 
| jani nikula | jani nikula | 8 | 4.76% | 1 | 16.67% | 
 | Total | 168 | 100.00% | 6 | 100.00% | 
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);
}
Contributors
 | Person | Tokens | Prop | Commits | CommitProp | 
| tomi valkeinen | tomi valkeinen | 54 | 100.00% | 1 | 100.00% | 
 | Total | 54 | 100.00% | 1 | 100.00% | 
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;
}
Contributors
 | Person | Tokens | Prop | Commits | CommitProp | 
| tomi valkeinen | tomi valkeinen | 178 | 94.18% | 4 | 66.67% | 
| jani nikula | jani nikula | 8 | 4.23% | 1 | 16.67% | 
| ville syrjala | ville syrjala | 3 | 1.59% | 1 | 16.67% | 
 | Total | 189 | 100.00% | 6 | 100.00% | 
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;
}
Contributors
 | Person | Tokens | Prop | Commits | CommitProp | 
| tomi valkeinen | tomi valkeinen | 170 | 91.40% | 1 | 33.33% | 
| jani nikula | jani nikula | 16 | 8.60% | 2 | 66.67% | 
 | Total | 186 | 100.00% | 3 | 100.00% | 
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;
}
Contributors
 | Person | Tokens | Prop | Commits | CommitProp | 
| tomi valkeinen | tomi valkeinen | 89 | 100.00% | 1 | 100.00% | 
 | Total | 89 | 100.00% | 1 | 100.00% | 
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;
}
Contributors
 | Person | Tokens | Prop | Commits | CommitProp | 
| tomi valkeinen | tomi valkeinen | 644 | 97.58% | 2 | 50.00% | 
| jani nikula | jani nikula | 16 | 2.42% | 2 | 50.00% | 
 | Total | 660 | 100.00% | 4 | 100.00% | 
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;
}
Contributors
 | Person | Tokens | Prop | Commits | CommitProp | 
| tomi valkeinen | tomi valkeinen | 122 | 93.85% | 1 | 50.00% | 
| jani nikula | jani nikula | 8 | 6.15% | 1 | 50.00% | 
 | Total | 130 | 100.00% | 2 | 100.00% | 
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;
}
Contributors
 | Person | Tokens | Prop | Commits | CommitProp | 
| tomi valkeinen | tomi valkeinen | 296 | 97.37% | 2 | 66.67% | 
| jani nikula | jani nikula | 8 | 2.63% | 1 | 33.33% | 
 | Total | 304 | 100.00% | 3 | 100.00% | 
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);
}
Contributors
 | Person | Tokens | Prop | Commits | CommitProp | 
| tomi valkeinen | tomi valkeinen | 55 | 98.21% | 1 | 50.00% | 
| ville syrjala | ville syrjala | 1 | 1.79% | 1 | 50.00% | 
 | Total | 56 | 100.00% | 2 | 100.00% | 
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;
}
Contributors
 | Person | Tokens | Prop | Commits | CommitProp | 
| tomi valkeinen | tomi valkeinen | 240 | 68.97% | 5 | 50.00% | 
| ville syrjala | ville syrjala | 100 | 28.74% | 4 | 40.00% | 
| jani nikula | jani nikula | 8 | 2.30% | 1 | 10.00% | 
 | Total | 348 | 100.00% | 10 | 100.00% | 
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);
}
Contributors
 | Person | Tokens | Prop | Commits | CommitProp | 
| tomi valkeinen | tomi valkeinen | 55 | 98.21% | 1 | 50.00% | 
| ville syrjala | ville syrjala | 1 | 1.79% | 1 | 50.00% | 
 | Total | 56 | 100.00% | 2 | 100.00% | 
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);
}
Contributors
 | Person | Tokens | Prop | Commits | CommitProp | 
| tomi valkeinen | tomi valkeinen | 55 | 98.21% | 1 | 50.00% | 
| ville syrjala | ville syrjala | 1 | 1.79% | 1 | 50.00% | 
 | Total | 56 | 100.00% | 2 | 100.00% | 
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);
}
Contributors
 | Person | Tokens | Prop | Commits | CommitProp | 
| tomi valkeinen | tomi valkeinen | 69 | 100.00% | 1 | 100.00% | 
 | Total | 69 | 100.00% | 1 | 100.00% | 
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;
}
Contributors
 | Person | Tokens | Prop | Commits | CommitProp | 
| tomi valkeinen | tomi valkeinen | 78 | 100.00% | 1 | 100.00% | 
 | Total | 78 | 100.00% | 1 | 100.00% | 
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;
}
Contributors
 | Person | Tokens | Prop | Commits | CommitProp | 
| tomi valkeinen | tomi valkeinen | 101 | 100.00% | 1 | 100.00% | 
 | Total | 101 | 100.00% | 1 | 100.00% | 
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]);
	}
}
Contributors
 | Person | Tokens | Prop | Commits | CommitProp | 
| tomi valkeinen | tomi valkeinen | 71 | 100.00% | 1 | 100.00% | 
 | Total | 71 | 100.00% | 1 | 100.00% | 
Overall Contributors
 | Person | Tokens | Prop | Commits | CommitProp | 
| tomi valkeinen | tomi valkeinen | 2614 | 92.99% | 8 | 53.33% | 
| ville syrjala | ville syrjala | 124 | 4.41% | 4 | 26.67% | 
| jani nikula | jani nikula | 72 | 2.56% | 2 | 13.33% | 
| peter ujfalusi | peter ujfalusi | 1 | 0.04% | 1 | 6.67% | 
 | Total | 2811 | 100.00% | 15 | 100.00% |