Commit b3e88137 authored by Thomas Zimmermann's avatar Thomas Zimmermann

fbdev: Warn on incorrect framebuffer access

Test in framebuffer read, write and drawing helpers if FBINFO_VIRTFB
has been set correctly. Framebuffers in I/O memory should only be
accessed with the architecture's respective helpers. Framebuffers
in system memory should be accessed with the regular load and
store operations. Presumably not all drivers get this right, so we
now warn about it.
Signed-off-by: default avatarThomas Zimmermann <tzimmermann@suse.de>
Reviewed-by: default avatarJavier Martinez Canillas <javierm@redhat.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20231127131655.4020-32-tzimmermann@suse.de
parent 33253d9e
...@@ -391,6 +391,9 @@ void cfb_copyarea(struct fb_info *p, const struct fb_copyarea *area) ...@@ -391,6 +391,9 @@ void cfb_copyarea(struct fb_info *p, const struct fb_copyarea *area)
if (p->state != FBINFO_STATE_RUNNING) if (p->state != FBINFO_STATE_RUNNING)
return; return;
if (p->flags & FBINFO_VIRTFB)
fb_warn_once(p, "Framebuffer is not in I/O address space.");
/* if the beginning of the target area might overlap with the end of /* if the beginning of the target area might overlap with the end of
the source area, be have to copy the area reverse. */ the source area, be have to copy the area reverse. */
if ((dy == sy && dx > sx) || (dy > sy)) { if ((dy == sy && dx > sx) || (dy > sy)) {
......
...@@ -287,6 +287,9 @@ void cfb_fillrect(struct fb_info *p, const struct fb_fillrect *rect) ...@@ -287,6 +287,9 @@ void cfb_fillrect(struct fb_info *p, const struct fb_fillrect *rect)
if (p->state != FBINFO_STATE_RUNNING) if (p->state != FBINFO_STATE_RUNNING)
return; return;
if (p->flags & FBINFO_VIRTFB)
fb_warn_once(p, "Framebuffer is not in I/O address space.");
if (p->fix.visual == FB_VISUAL_TRUECOLOR || if (p->fix.visual == FB_VISUAL_TRUECOLOR ||
p->fix.visual == FB_VISUAL_DIRECTCOLOR ) p->fix.visual == FB_VISUAL_DIRECTCOLOR )
fg = ((u32 *) (p->pseudo_palette))[rect->color]; fg = ((u32 *) (p->pseudo_palette))[rect->color];
......
...@@ -326,6 +326,9 @@ void cfb_imageblit(struct fb_info *p, const struct fb_image *image) ...@@ -326,6 +326,9 @@ void cfb_imageblit(struct fb_info *p, const struct fb_image *image)
if (p->state != FBINFO_STATE_RUNNING) if (p->state != FBINFO_STATE_RUNNING)
return; return;
if (p->flags & FBINFO_VIRTFB)
fb_warn_once(p, "Framebuffer is not in I/O address space.");
bitstart = (dy * p->fix.line_length * 8) + (dx * bpp); bitstart = (dy * p->fix.line_length * 8) + (dx * bpp);
start_index = bitstart & (32 - 1); start_index = bitstart & (32 - 1);
pitch_index = (p->fix.line_length & (bpl - 1)) * 8; pitch_index = (p->fix.line_length & (bpl - 1)) * 8;
......
...@@ -12,6 +12,9 @@ ssize_t fb_io_read(struct fb_info *info, char __user *buf, size_t count, loff_t ...@@ -12,6 +12,9 @@ ssize_t fb_io_read(struct fb_info *info, char __user *buf, size_t count, loff_t
int c, cnt = 0, err = 0; int c, cnt = 0, err = 0;
unsigned long total_size, trailing; unsigned long total_size, trailing;
if (info->flags & FBINFO_VIRTFB)
fb_warn_once(info, "Framebuffer is not in I/O address space.");
if (!info->screen_base) if (!info->screen_base)
return -ENODEV; return -ENODEV;
...@@ -73,6 +76,9 @@ ssize_t fb_io_write(struct fb_info *info, const char __user *buf, size_t count, ...@@ -73,6 +76,9 @@ ssize_t fb_io_write(struct fb_info *info, const char __user *buf, size_t count,
int c, cnt = 0, err = 0; int c, cnt = 0, err = 0;
unsigned long total_size, trailing; unsigned long total_size, trailing;
if (info->flags & FBINFO_VIRTFB)
fb_warn_once(info, "Framebuffer is not in I/O address space.");
if (!info->screen_base) if (!info->screen_base)
return -ENODEV; return -ENODEV;
...@@ -138,6 +144,9 @@ int fb_io_mmap(struct fb_info *info, struct vm_area_struct *vma) ...@@ -138,6 +144,9 @@ int fb_io_mmap(struct fb_info *info, struct vm_area_struct *vma)
u32 len = info->fix.smem_len; u32 len = info->fix.smem_len;
unsigned long mmio_pgoff = PAGE_ALIGN((start & ~PAGE_MASK) + len) >> PAGE_SHIFT; unsigned long mmio_pgoff = PAGE_ALIGN((start & ~PAGE_MASK) + len) >> PAGE_SHIFT;
if (info->flags & FBINFO_VIRTFB)
fb_warn_once(info, "Framebuffer is not in I/O address space.");
/* /*
* This can be either the framebuffer mapping, or if pgoff points * This can be either the framebuffer mapping, or if pgoff points
* past it, the mmio mapping. * past it, the mmio mapping.
......
...@@ -22,6 +22,9 @@ ssize_t fb_sys_read(struct fb_info *info, char __user *buf, size_t count, ...@@ -22,6 +22,9 @@ ssize_t fb_sys_read(struct fb_info *info, char __user *buf, size_t count,
unsigned long total_size, c; unsigned long total_size, c;
ssize_t ret; ssize_t ret;
if (!(info->flags & FBINFO_VIRTFB))
fb_warn_once(info, "Framebuffer is not in virtual address space.");
if (!info->screen_buffer) if (!info->screen_buffer)
return -ENODEV; return -ENODEV;
...@@ -64,6 +67,9 @@ ssize_t fb_sys_write(struct fb_info *info, const char __user *buf, ...@@ -64,6 +67,9 @@ ssize_t fb_sys_write(struct fb_info *info, const char __user *buf,
unsigned long total_size, c; unsigned long total_size, c;
size_t ret; size_t ret;
if (!(info->flags & FBINFO_VIRTFB))
fb_warn_once(info, "Framebuffer is not in virtual address space.");
if (!info->screen_buffer) if (!info->screen_buffer)
return -ENODEV; return -ENODEV;
......
...@@ -324,6 +324,9 @@ void sys_copyarea(struct fb_info *p, const struct fb_copyarea *area) ...@@ -324,6 +324,9 @@ void sys_copyarea(struct fb_info *p, const struct fb_copyarea *area)
if (p->state != FBINFO_STATE_RUNNING) if (p->state != FBINFO_STATE_RUNNING)
return; return;
if (!(p->flags & FBINFO_VIRTFB))
fb_warn_once(p, "Framebuffer is not in virtual address space.");
/* if the beginning of the target area might overlap with the end of /* if the beginning of the target area might overlap with the end of
the source area, be have to copy the area reverse. */ the source area, be have to copy the area reverse. */
if ((dy == sy && dx > sx) || (dy > sy)) { if ((dy == sy && dx > sx) || (dy > sy)) {
......
...@@ -242,6 +242,9 @@ void sys_fillrect(struct fb_info *p, const struct fb_fillrect *rect) ...@@ -242,6 +242,9 @@ void sys_fillrect(struct fb_info *p, const struct fb_fillrect *rect)
if (p->state != FBINFO_STATE_RUNNING) if (p->state != FBINFO_STATE_RUNNING)
return; return;
if (!(p->flags & FBINFO_VIRTFB))
fb_warn_once(p, "Framebuffer is not in virtual address space.");
if (p->fix.visual == FB_VISUAL_TRUECOLOR || if (p->fix.visual == FB_VISUAL_TRUECOLOR ||
p->fix.visual == FB_VISUAL_DIRECTCOLOR ) p->fix.visual == FB_VISUAL_DIRECTCOLOR )
fg = ((u32 *) (p->pseudo_palette))[rect->color]; fg = ((u32 *) (p->pseudo_palette))[rect->color];
......
...@@ -296,6 +296,9 @@ void sys_imageblit(struct fb_info *p, const struct fb_image *image) ...@@ -296,6 +296,9 @@ void sys_imageblit(struct fb_info *p, const struct fb_image *image)
if (p->state != FBINFO_STATE_RUNNING) if (p->state != FBINFO_STATE_RUNNING)
return; return;
if (!(p->flags & FBINFO_VIRTFB))
fb_warn_once(p, "Framebuffer is not in virtual address space.");
bitstart = (dy * p->fix.line_length * 8) + (dx * bpp); bitstart = (dy * p->fix.line_length * 8) + (dx * bpp);
start_index = bitstart & (32 - 1); start_index = bitstart & (32 - 1);
pitch_index = (p->fix.line_length & (bpl - 1)) * 8; pitch_index = (p->fix.line_length & (bpl - 1)) * 8;
......
...@@ -849,7 +849,10 @@ static inline bool fb_modesetting_disabled(const char *drvname) ...@@ -849,7 +849,10 @@ static inline bool fb_modesetting_disabled(const char *drvname)
} }
#endif #endif
/* Convenience logging macros */ /*
* Convenience logging macros
*/
#define fb_err(fb_info, fmt, ...) \ #define fb_err(fb_info, fmt, ...) \
pr_err("fb%d: " fmt, (fb_info)->node, ##__VA_ARGS__) pr_err("fb%d: " fmt, (fb_info)->node, ##__VA_ARGS__)
#define fb_notice(info, fmt, ...) \ #define fb_notice(info, fmt, ...) \
...@@ -861,4 +864,7 @@ static inline bool fb_modesetting_disabled(const char *drvname) ...@@ -861,4 +864,7 @@ static inline bool fb_modesetting_disabled(const char *drvname)
#define fb_dbg(fb_info, fmt, ...) \ #define fb_dbg(fb_info, fmt, ...) \
pr_debug("fb%d: " fmt, (fb_info)->node, ##__VA_ARGS__) pr_debug("fb%d: " fmt, (fb_info)->node, ##__VA_ARGS__)
#define fb_warn_once(fb_info, fmt, ...) \
pr_warn_once("fb%d: " fmt, (fb_info)->node, ##__VA_ARGS__)
#endif /* _LINUX_FB_H */ #endif /* _LINUX_FB_H */
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment