- patches.suse/slab-handle-memoryless-nodes-v2a.patch: Refresh.
[linux-flexiantxendom0-3.2.10.git] / drivers / video / omap2 / omapfb / omapfb-sysfs.c
1 /*
2  * linux/drivers/video/omap2/omapfb-sysfs.c
3  *
4  * Copyright (C) 2008 Nokia Corporation
5  * Author: Tomi Valkeinen <tomi.valkeinen@nokia.com>
6  *
7  * Some code and ideas taken from drivers/video/omap/ driver
8  * by Imre Deak.
9  *
10  * This program is free software; you can redistribute it and/or modify it
11  * under the terms of the GNU General Public License version 2 as published by
12  * the Free Software Foundation.
13  *
14  * This program is distributed in the hope that it will be useful, but WITHOUT
15  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
16  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for
17  * more details.
18  *
19  * You should have received a copy of the GNU General Public License along with
20  * this program.  If not, see <http://www.gnu.org/licenses/>.
21  */
22
23 #include <linux/fb.h>
24 #include <linux/sysfs.h>
25 #include <linux/device.h>
26 #include <linux/uaccess.h>
27 #include <linux/platform_device.h>
28 #include <linux/kernel.h>
29 #include <linux/mm.h>
30 #include <linux/omapfb.h>
31
32 #include <plat/display.h>
33 #include <plat/vrfb.h>
34
35 #include "omapfb.h"
36
37 static ssize_t show_rotate_type(struct device *dev,
38                 struct device_attribute *attr, char *buf)
39 {
40         struct fb_info *fbi = dev_get_drvdata(dev);
41         struct omapfb_info *ofbi = FB2OFB(fbi);
42
43         return snprintf(buf, PAGE_SIZE, "%d\n", ofbi->rotation_type);
44 }
45
46 static ssize_t store_rotate_type(struct device *dev,
47                 struct device_attribute *attr,
48                 const char *buf, size_t count)
49 {
50         struct fb_info *fbi = dev_get_drvdata(dev);
51         struct omapfb_info *ofbi = FB2OFB(fbi);
52         enum omap_dss_rotation_type rot_type;
53         int r;
54
55         rot_type = simple_strtoul(buf, NULL, 0);
56
57         if (rot_type != OMAP_DSS_ROT_DMA && rot_type != OMAP_DSS_ROT_VRFB)
58                 return -EINVAL;
59
60         lock_fb_info(fbi);
61
62         r = 0;
63         if (rot_type == ofbi->rotation_type)
64                 goto out;
65
66         if (ofbi->region.size) {
67                 r = -EBUSY;
68                 goto out;
69         }
70
71         ofbi->rotation_type = rot_type;
72
73         /*
74          * Since the VRAM for this FB is not allocated at the moment we don't
75          * need to do any further parameter checking at this point.
76          */
77 out:
78         unlock_fb_info(fbi);
79
80         return r ? r : count;
81 }
82
83
84 static ssize_t show_mirror(struct device *dev,
85                 struct device_attribute *attr, char *buf)
86 {
87         struct fb_info *fbi = dev_get_drvdata(dev);
88         struct omapfb_info *ofbi = FB2OFB(fbi);
89
90         return snprintf(buf, PAGE_SIZE, "%d\n", ofbi->mirror);
91 }
92
93 static ssize_t store_mirror(struct device *dev,
94                 struct device_attribute *attr,
95                 const char *buf, size_t count)
96 {
97         struct fb_info *fbi = dev_get_drvdata(dev);
98         struct omapfb_info *ofbi = FB2OFB(fbi);
99         bool mirror;
100         int r;
101         struct fb_var_screeninfo new_var;
102
103         mirror = simple_strtoul(buf, NULL, 0);
104
105         if (mirror != 0 && mirror != 1)
106                 return -EINVAL;
107
108         lock_fb_info(fbi);
109
110         ofbi->mirror = mirror;
111
112         memcpy(&new_var, &fbi->var, sizeof(new_var));
113         r = check_fb_var(fbi, &new_var);
114         if (r)
115                 goto out;
116         memcpy(&fbi->var, &new_var, sizeof(fbi->var));
117
118         set_fb_fix(fbi);
119
120         r = omapfb_apply_changes(fbi, 0);
121         if (r)
122                 goto out;
123
124         r = count;
125 out:
126         unlock_fb_info(fbi);
127
128         return r;
129 }
130
131 static ssize_t show_overlays(struct device *dev,
132                 struct device_attribute *attr, char *buf)
133 {
134         struct fb_info *fbi = dev_get_drvdata(dev);
135         struct omapfb_info *ofbi = FB2OFB(fbi);
136         struct omapfb2_device *fbdev = ofbi->fbdev;
137         ssize_t l = 0;
138         int t;
139
140         omapfb_lock(fbdev);
141         lock_fb_info(fbi);
142
143         for (t = 0; t < ofbi->num_overlays; t++) {
144                 struct omap_overlay *ovl = ofbi->overlays[t];
145                 int ovlnum;
146
147                 for (ovlnum = 0; ovlnum < fbdev->num_overlays; ++ovlnum)
148                         if (ovl == fbdev->overlays[ovlnum])
149                                 break;
150
151                 l += snprintf(buf + l, PAGE_SIZE - l, "%s%d",
152                                 t == 0 ? "" : ",", ovlnum);
153         }
154
155         l += snprintf(buf + l, PAGE_SIZE - l, "\n");
156
157         unlock_fb_info(fbi);
158         omapfb_unlock(fbdev);
159
160         return l;
161 }
162
163 static struct omapfb_info *get_overlay_fb(struct omapfb2_device *fbdev,
164                 struct omap_overlay *ovl)
165 {
166         int i, t;
167
168         for (i = 0; i < fbdev->num_fbs; i++) {
169                 struct omapfb_info *ofbi = FB2OFB(fbdev->fbs[i]);
170
171                 for (t = 0; t < ofbi->num_overlays; t++) {
172                         if (ofbi->overlays[t] == ovl)
173                                 return ofbi;
174                 }
175         }
176
177         return NULL;
178 }
179
180 static ssize_t store_overlays(struct device *dev, struct device_attribute *attr,
181                 const char *buf, size_t count)
182 {
183         struct fb_info *fbi = dev_get_drvdata(dev);
184         struct omapfb_info *ofbi = FB2OFB(fbi);
185         struct omapfb2_device *fbdev = ofbi->fbdev;
186         struct omap_overlay *ovls[OMAPFB_MAX_OVL_PER_FB];
187         struct omap_overlay *ovl;
188         int num_ovls, r, i;
189         int len;
190         bool added = false;
191
192         num_ovls = 0;
193
194         len = strlen(buf);
195         if (buf[len - 1] == '\n')
196                 len = len - 1;
197
198         omapfb_lock(fbdev);
199         lock_fb_info(fbi);
200
201         if (len > 0) {
202                 char *p = (char *)buf;
203                 int ovlnum;
204
205                 while (p < buf + len) {
206                         int found;
207                         if (num_ovls == OMAPFB_MAX_OVL_PER_FB) {
208                                 r = -EINVAL;
209                                 goto out;
210                         }
211
212                         ovlnum = simple_strtoul(p, &p, 0);
213                         if (ovlnum > fbdev->num_overlays) {
214                                 r = -EINVAL;
215                                 goto out;
216                         }
217
218                         found = 0;
219                         for (i = 0; i < num_ovls; ++i) {
220                                 if (ovls[i] == fbdev->overlays[ovlnum]) {
221                                         found = 1;
222                                         break;
223                                 }
224                         }
225
226                         if (!found)
227                                 ovls[num_ovls++] = fbdev->overlays[ovlnum];
228
229                         p++;
230                 }
231         }
232
233         for (i = 0; i < num_ovls; ++i) {
234                 struct omapfb_info *ofbi2 = get_overlay_fb(fbdev, ovls[i]);
235                 if (ofbi2 && ofbi2 != ofbi) {
236                         dev_err(fbdev->dev, "overlay already in use\n");
237                         r = -EINVAL;
238                         goto out;
239                 }
240         }
241
242         /* detach unused overlays */
243         for (i = 0; i < ofbi->num_overlays; ++i) {
244                 int t, found;
245
246                 ovl = ofbi->overlays[i];
247
248                 found = 0;
249
250                 for (t = 0; t < num_ovls; ++t) {
251                         if (ovl == ovls[t]) {
252                                 found = 1;
253                                 break;
254                         }
255                 }
256
257                 if (found)
258                         continue;
259
260                 DBG("detaching %d\n", ofbi->overlays[i]->id);
261
262                 omapfb_overlay_enable(ovl, 0);
263
264                 if (ovl->manager)
265                         ovl->manager->apply(ovl->manager);
266
267                 for (t = i + 1; t < ofbi->num_overlays; t++) {
268                         ofbi->rotation[t-1] = ofbi->rotation[t];
269                         ofbi->overlays[t-1] = ofbi->overlays[t];
270                 }
271
272                 ofbi->num_overlays--;
273                 i--;
274         }
275
276         for (i = 0; i < num_ovls; ++i) {
277                 int t, found;
278
279                 ovl = ovls[i];
280
281                 found = 0;
282
283                 for (t = 0; t < ofbi->num_overlays; ++t) {
284                         if (ovl == ofbi->overlays[t]) {
285                                 found = 1;
286                                 break;
287                         }
288                 }
289
290                 if (found)
291                         continue;
292                 ofbi->rotation[ofbi->num_overlays] = 0;
293                 ofbi->overlays[ofbi->num_overlays++] = ovl;
294
295                 added = true;
296         }
297
298         if (added) {
299                 r = omapfb_apply_changes(fbi, 0);
300                 if (r)
301                         goto out;
302         }
303
304         r = count;
305 out:
306         unlock_fb_info(fbi);
307         omapfb_unlock(fbdev);
308
309         return r;
310 }
311
312 static ssize_t show_overlays_rotate(struct device *dev,
313                 struct device_attribute *attr, char *buf)
314 {
315         struct fb_info *fbi = dev_get_drvdata(dev);
316         struct omapfb_info *ofbi = FB2OFB(fbi);
317         ssize_t l = 0;
318         int t;
319
320         lock_fb_info(fbi);
321
322         for (t = 0; t < ofbi->num_overlays; t++) {
323                 l += snprintf(buf + l, PAGE_SIZE - l, "%s%d",
324                                 t == 0 ? "" : ",", ofbi->rotation[t]);
325         }
326
327         l += snprintf(buf + l, PAGE_SIZE - l, "\n");
328
329         unlock_fb_info(fbi);
330
331         return l;
332 }
333
334 static ssize_t store_overlays_rotate(struct device *dev,
335                 struct device_attribute *attr, const char *buf, size_t count)
336 {
337         struct fb_info *fbi = dev_get_drvdata(dev);
338         struct omapfb_info *ofbi = FB2OFB(fbi);
339         int num_ovls = 0, r, i;
340         int len;
341         bool changed = false;
342         u8 rotation[OMAPFB_MAX_OVL_PER_FB];
343
344         len = strlen(buf);
345         if (buf[len - 1] == '\n')
346                 len = len - 1;
347
348         lock_fb_info(fbi);
349
350         if (len > 0) {
351                 char *p = (char *)buf;
352
353                 while (p < buf + len) {
354                         int rot;
355
356                         if (num_ovls == ofbi->num_overlays) {
357                                 r = -EINVAL;
358                                 goto out;
359                         }
360
361                         rot = simple_strtoul(p, &p, 0);
362                         if (rot < 0 || rot > 3) {
363                                 r = -EINVAL;
364                                 goto out;
365                         }
366
367                         if (ofbi->rotation[num_ovls] != rot)
368                                 changed = true;
369
370                         rotation[num_ovls++] = rot;
371
372                         p++;
373                 }
374         }
375
376         if (num_ovls != ofbi->num_overlays) {
377                 r = -EINVAL;
378                 goto out;
379         }
380
381         if (changed) {
382                 for (i = 0; i < num_ovls; ++i)
383                         ofbi->rotation[i] = rotation[i];
384
385                 r = omapfb_apply_changes(fbi, 0);
386                 if (r)
387                         goto out;
388
389                 /* FIXME error handling? */
390         }
391
392         r = count;
393 out:
394         unlock_fb_info(fbi);
395
396         return r;
397 }
398
399 static ssize_t show_size(struct device *dev,
400                 struct device_attribute *attr, char *buf)
401 {
402         struct fb_info *fbi = dev_get_drvdata(dev);
403         struct omapfb_info *ofbi = FB2OFB(fbi);
404
405         return snprintf(buf, PAGE_SIZE, "%lu\n", ofbi->region.size);
406 }
407
408 static ssize_t store_size(struct device *dev, struct device_attribute *attr,
409                 const char *buf, size_t count)
410 {
411         struct fb_info *fbi = dev_get_drvdata(dev);
412         struct omapfb_info *ofbi = FB2OFB(fbi);
413         unsigned long size;
414         int r;
415         int i;
416
417         size = PAGE_ALIGN(simple_strtoul(buf, NULL, 0));
418
419         lock_fb_info(fbi);
420
421         for (i = 0; i < ofbi->num_overlays; i++) {
422                 if (ofbi->overlays[i]->info.enabled) {
423                         r = -EBUSY;
424                         goto out;
425                 }
426         }
427
428         if (size != ofbi->region.size) {
429                 r = omapfb_realloc_fbmem(fbi, size, ofbi->region.type);
430                 if (r) {
431                         dev_err(dev, "realloc fbmem failed\n");
432                         goto out;
433                 }
434         }
435
436         r = count;
437 out:
438         unlock_fb_info(fbi);
439
440         return r;
441 }
442
443 static ssize_t show_phys(struct device *dev,
444                 struct device_attribute *attr, char *buf)
445 {
446         struct fb_info *fbi = dev_get_drvdata(dev);
447         struct omapfb_info *ofbi = FB2OFB(fbi);
448
449         return snprintf(buf, PAGE_SIZE, "%0x\n", ofbi->region.paddr);
450 }
451
452 static ssize_t show_virt(struct device *dev,
453                 struct device_attribute *attr, char *buf)
454 {
455         struct fb_info *fbi = dev_get_drvdata(dev);
456         struct omapfb_info *ofbi = FB2OFB(fbi);
457
458         return snprintf(buf, PAGE_SIZE, "%p\n", ofbi->region.vaddr);
459 }
460
461 static struct device_attribute omapfb_attrs[] = {
462         __ATTR(rotate_type, S_IRUGO | S_IWUSR, show_rotate_type,
463                         store_rotate_type),
464         __ATTR(mirror, S_IRUGO | S_IWUSR, show_mirror, store_mirror),
465         __ATTR(size, S_IRUGO | S_IWUSR, show_size, store_size),
466         __ATTR(overlays, S_IRUGO | S_IWUSR, show_overlays, store_overlays),
467         __ATTR(overlays_rotate, S_IRUGO | S_IWUSR, show_overlays_rotate,
468                         store_overlays_rotate),
469         __ATTR(phys_addr, S_IRUGO, show_phys, NULL),
470         __ATTR(virt_addr, S_IRUGO, show_virt, NULL),
471 };
472
473 int omapfb_create_sysfs(struct omapfb2_device *fbdev)
474 {
475         int i;
476         int r;
477
478         DBG("create sysfs for fbs\n");
479         for (i = 0; i < fbdev->num_fbs; i++) {
480                 int t;
481                 for (t = 0; t < ARRAY_SIZE(omapfb_attrs); t++) {
482                         r = device_create_file(fbdev->fbs[i]->dev,
483                                         &omapfb_attrs[t]);
484
485                         if (r) {
486                                 dev_err(fbdev->dev, "failed to create sysfs "
487                                                 "file\n");
488                                 return r;
489                         }
490                 }
491         }
492
493         return 0;
494 }
495
496 void omapfb_remove_sysfs(struct omapfb2_device *fbdev)
497 {
498         int i, t;
499
500         DBG("remove sysfs for fbs\n");
501         for (i = 0; i < fbdev->num_fbs; i++) {
502                 for (t = 0; t < ARRAY_SIZE(omapfb_attrs); t++)
503                         device_remove_file(fbdev->fbs[i]->dev,
504                                         &omapfb_attrs[t]);
505         }
506 }
507