| // SPDX-License-Identifier: GPL-2.0 |
| |
| #include <linux/fb.h> |
| #include <linux/module.h> |
| #include <linux/uaccess.h> |
| |
| ssize_t fb_io_read(struct fb_info *info, char __user *buf, size_t count, loff_t *ppos) |
| { |
| unsigned long p = *ppos; |
| u8 *buffer, *dst; |
| u8 __iomem *src; |
| int c, cnt = 0, err = 0; |
| 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) |
| return -ENODEV; |
| |
| total_size = info->screen_size; |
| |
| if (total_size == 0) |
| total_size = info->fix.smem_len; |
| |
| if (p >= total_size) |
| return 0; |
| |
| if (count >= total_size) |
| count = total_size; |
| |
| if (count + p > total_size) |
| count = total_size - p; |
| |
| buffer = kmalloc((count > PAGE_SIZE) ? PAGE_SIZE : count, |
| GFP_KERNEL); |
| if (!buffer) |
| return -ENOMEM; |
| |
| src = (u8 __iomem *) (info->screen_base + p); |
| |
| if (info->fbops->fb_sync) |
| info->fbops->fb_sync(info); |
| |
| while (count) { |
| c = (count > PAGE_SIZE) ? PAGE_SIZE : count; |
| dst = buffer; |
| fb_memcpy_fromio(dst, src, c); |
| dst += c; |
| src += c; |
| |
| trailing = copy_to_user(buf, buffer, c); |
| if (trailing == c) { |
| err = -EFAULT; |
| break; |
| } |
| c -= trailing; |
| |
| *ppos += c; |
| buf += c; |
| cnt += c; |
| count -= c; |
| } |
| |
| kfree(buffer); |
| |
| return cnt ? cnt : err; |
| } |
| EXPORT_SYMBOL(fb_io_read); |
| |
| ssize_t fb_io_write(struct fb_info *info, const char __user *buf, size_t count, loff_t *ppos) |
| { |
| unsigned long p = *ppos; |
| u8 *buffer, *src; |
| u8 __iomem *dst; |
| int c, cnt = 0, err = 0; |
| 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) |
| return -ENODEV; |
| |
| total_size = info->screen_size; |
| |
| if (total_size == 0) |
| total_size = info->fix.smem_len; |
| |
| if (p > total_size) |
| return -EFBIG; |
| |
| if (count > total_size) { |
| err = -EFBIG; |
| count = total_size; |
| } |
| |
| if (count + p > total_size) { |
| if (!err) |
| err = -ENOSPC; |
| |
| count = total_size - p; |
| } |
| |
| buffer = kmalloc((count > PAGE_SIZE) ? PAGE_SIZE : count, |
| GFP_KERNEL); |
| if (!buffer) |
| return -ENOMEM; |
| |
| dst = (u8 __iomem *) (info->screen_base + p); |
| |
| if (info->fbops->fb_sync) |
| info->fbops->fb_sync(info); |
| |
| while (count) { |
| c = (count > PAGE_SIZE) ? PAGE_SIZE : count; |
| src = buffer; |
| |
| trailing = copy_from_user(src, buf, c); |
| if (trailing == c) { |
| err = -EFAULT; |
| break; |
| } |
| c -= trailing; |
| |
| fb_memcpy_toio(dst, src, c); |
| dst += c; |
| src += c; |
| *ppos += c; |
| buf += c; |
| cnt += c; |
| count -= c; |
| } |
| |
| kfree(buffer); |
| |
| return (cnt) ? cnt : err; |
| } |
| EXPORT_SYMBOL(fb_io_write); |
| |
| int fb_io_mmap(struct fb_info *info, struct vm_area_struct *vma) |
| { |
| unsigned long start = info->fix.smem_start; |
| u32 len = info->fix.smem_len; |
| 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 |
| * past it, the mmio mapping. |
| */ |
| if (vma->vm_pgoff >= mmio_pgoff) { |
| if (info->var.accel_flags) |
| return -EINVAL; |
| |
| vma->vm_pgoff -= mmio_pgoff; |
| start = info->fix.mmio_start; |
| len = info->fix.mmio_len; |
| } |
| |
| vma->vm_page_prot = vm_get_page_prot(vma->vm_flags); |
| vma->vm_page_prot = pgprot_framebuffer(vma->vm_page_prot, vma->vm_start, |
| vma->vm_end, start); |
| |
| return vm_iomap_memory(vma, start, len); |
| } |
| EXPORT_SYMBOL(fb_io_mmap); |
| |
| MODULE_DESCRIPTION("Fbdev helpers for framebuffers in I/O memory"); |
| MODULE_LICENSE("GPL"); |