419 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			419 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  *  linux/drivers/video/console/fbcondecor.c -- Framebuffer console decorations+ *
 | |
|  *  Copyright (C) 2004 Michal Januszewski <spock@gentoo.org>
 | |
|  *
 | |
|  *  Code based upon "Bootsplash" (C) 2001-2003
 | |
|  *       Volker Poplawski <volker@poplawski.de>,
 | |
|  *       Stefan Reinauer <stepan@suse.de>,
 | |
|  *       Steffen Winterfeldt <snwint@suse.de>,
 | |
|  *       Michael Schroeder <mls@suse.de>,
 | |
|  *       Ken Wimer <wimer@suse.de>.
 | |
|  *
 | |
|  *  This file is subject to the terms and conditions of the GNU General Public
 | |
|  *  License.  See the file COPYING in the main directory of this archive for
 | |
|  *  more details.
 | |
|  *
 | |
|  */
 | |
| #include <linux/module.h>
 | |
| #include <linux/kernel.h>
 | |
| #include <linux/string.h>
 | |
| #include <linux/types.h>
 | |
| #include <linux/fb.h>
 | |
| #include <linux/vt_kern.h>
 | |
| #include <linux/vmalloc.h>
 | |
| #include <linux/unistd.h>
 | |
| #include <linux/syscalls.h>
 | |
| #include <linux/init.h>
 | |
| #include <linux/proc_fs.h>
 | |
| #include <linux/workqueue.h>
 | |
| #include <linux/kmod.h>
 | |
| #include <linux/miscdevice.h>
 | |
| #include <linux/device.h>
 | |
| #include <linux/fs.h>
 | |
| 
 | |
| #include <asm/uaccess.h>
 | |
| #include <asm/irq.h>
 | |
| #include <asm/system.h>
 | |
| 
 | |
| #include "fbcon.h"
 | |
| #include "fbcondecor.h"
 | |
| 
 | |
| extern signed char con2fb_map[];
 | |
| static int fbcon_decor_enable(struct vc_data *vc);
 | |
| char fbcon_decor_path[KMOD_PATH_LEN] = "/sbin/fbcondecor_helper";
 | |
| static int initialized = 0;
 | |
| 
 | |
| int fbcon_decor_call_helper(char* cmd, unsigned short vc)
 | |
| {
 | |
|  char *envp[] = {
 | |
|    "HOME=/",
 | |
|    "PATH=/sbin:/bin",
 | |
|    NULL
 | |
|  };
 | |
| 
 | |
|  char tfb[5];
 | |
|  char tcons[5];
 | |
|  unsigned char fb = (int) con2fb_map[vc];
 | |
| 
 | |
|  char *argv[] = {
 | |
|    fbcon_decor_path,
 | |
|    "2",
 | |
|    cmd,
 | |
|    tcons,
 | |
|    tfb,
 | |
|    vc_cons[vc].d->vc_decor.theme,
 | |
|    NULL
 | |
|  };
 | |
| 
 | |
|  snprintf(tfb,5,"%d",fb);
 | |
|  snprintf(tcons,5,"%d",vc);
 | |
| 
 | |
|  return call_usermodehelper(fbcon_decor_path, argv, envp, 1);
 | |
| }
 | |
| 
 | |
| /* Disables fbcondecor on a virtual console; called with console sem held. */
 | |
| int fbcon_decor_disable(struct vc_data *vc, unsigned char redraw)
 | |
| {
 | |
|  struct fb_info* info;
 | |
| 
 | |
|  if (!vc->vc_decor.state)
 | |
|    return -EINVAL;
 | |
| 
 | |
|  info = registered_fb[(int) con2fb_map[vc->vc_num]];
 | |
| 
 | |
|  if (info == NULL)
 | |
|    return -EINVAL;
 | |
| 
 | |
|  vc->vc_decor.state = 0;
 | |
|  vc_resize(vc, info->var.xres / vc->vc_font.width,
 | |
|      info->var.yres / vc->vc_font.height);
 | |
| 
 | |
|  if (fg_console == vc->vc_num && redraw) {
 | |
|    redraw_screen(vc, 0);
 | |
|    update_region(vc, vc->vc_origin +
 | |
|            vc->vc_size_row * vc->vc_top,
 | |
|            vc->vc_size_row * (vc->vc_bottom - vc->vc_top) / 2);
 | |
|  }
 | |
| 
 | |
|  printk(KERN_INFO "fbcondecor: switched decor state to 'off' on console %d\n",
 | |
|       vc->vc_num);
 | |
| 
 | |
|  return 0;
 | |
| }
 | |
| 
 | |
| /* Enables fbcondecor on a virtual console; called with console sem held. */
 | |
| static int fbcon_decor_enable(struct vc_data *vc)
 | |
| {
 | |
|  struct fb_info* info;
 | |
| 
 | |
|  info = registered_fb[(int) con2fb_map[vc->vc_num]];
 | |
| 
 | |
|  if (vc->vc_decor.twidth == 0 || vc->vc_decor.theight == 0 ||
 | |
|      info == NULL || vc->vc_decor.state || (!info->bgdecor.data &&
 | |
|      vc->vc_num == fg_console))
 | |
|    return -EINVAL;
 | |
| 
 | |
|  vc->vc_decor.state = 1;
 | |
|  vc_resize(vc, vc->vc_decor.twidth / vc->vc_font.width,
 | |
|      vc->vc_decor.theight / vc->vc_font.height);
 | |
| 
 | |
|  if (fg_console == vc->vc_num) {
 | |
|    redraw_screen(vc, 0);
 | |
|    update_region(vc, vc->vc_origin +
 | |
|            vc->vc_size_row * vc->vc_top,
 | |
|            vc->vc_size_row * (vc->vc_bottom - vc->vc_top) / 2);
 | |
|    fbcon_decor_clear_margins(vc, info, 0);
 | |
|  }
 | |
| 
 | |
|  printk(KERN_INFO "fbcondecor: switched decor state to 'on' on console %d\n",
 | |
|       vc->vc_num);
 | |
| 
 | |
|  return 0;
 | |
| }
 | |
| 
 | |
| static inline int fbcon_decor_ioctl_dosetstate(struct vc_data *vc, unsigned int __user* state, unsigned char origin)
 | |
| {
 | |
|  int tmp, ret;
 | |
|  if (get_user(tmp, state))
 | |
|    return -EFAULT;
 | |
| 
 | |
|  if (origin == FBCON_DECOR_IO_ORIG_USER)
 | |
|    acquire_console_sem();
 | |
|  if (!tmp)
 | |
|    ret = fbcon_decor_disable(vc, 1);
 | |
|  else
 | |
|    ret = fbcon_decor_enable(vc);
 | |
|  if (origin == FBCON_DECOR_IO_ORIG_USER)
 | |
|    release_console_sem();
 | |
| 
 | |
|  return ret;
 | |
| }
 | |
| 
 | |
| static inline int fbcon_decor_ioctl_dogetstate(struct vc_data *vc, unsigned int __user *state)
 | |
| {
 | |
|  return put_user(vc->vc_decor.state, (unsigned int __user*) state);
 | |
| }
 | |
| 
 | |
| static int fbcon_decor_ioctl_dosetcfg(struct vc_data *vc, struct vc_decor __user *arg, unsigned char origin)
 | |
| {
 | |
|  struct vc_decor cfg;
 | |
|  struct fb_info *info;
 | |
|  int len;
 | |
|  char *tmp;
 | |
| 
 | |
|  info = registered_fb[(int) con2fb_map[vc->vc_num]];
 | |
| 
 | |
|  if (copy_from_user(&cfg, arg, sizeof(struct vc_decor)))
 | |
|    return -EFAULT;
 | |
|  if (info == NULL || !cfg.twidth || !cfg.theight ||
 | |
|      cfg.tx + cfg.twidth  > info->var.xres ||
 | |
|      cfg.ty + cfg.theight > info->var.yres)
 | |
|    return -EINVAL;
 | |
| 
 | |
|  len = strlen_user(cfg.theme);
 | |
|  if (!len || len > FBCON_DECOR_THEME_LEN)
 | |
|    return -EINVAL;
 | |
|  tmp = kmalloc(len, GFP_KERNEL);
 | |
|  if (!tmp)
 | |
|    return -ENOMEM;
 | |
|  if (copy_from_user(tmp, (void __user *)cfg.theme, len))
 | |
|    return -EFAULT;
 | |
|  cfg.theme = tmp;
 | |
|  cfg.state = 0;
 | |
| 
 | |
|  /* If this ioctl is a response to a request from kernel, the console sem
 | |
|   * is already held; we also don't need to disable decor because either the
 | |
|   * new config and background picture will be successfully loaded, and the
 | |
|   * decor will stay on, or in case of a failure it'll be turned off in fbcon. */
 | |
|  if (origin == FBCON_DECOR_IO_ORIG_USER) {
 | |
|    acquire_console_sem();
 | |
|    if (vc->vc_decor.state)
 | |
|      fbcon_decor_disable(vc, 1);
 | |
|  }
 | |
| 
 | |
|  if (vc->vc_decor.theme)
 | |
|    kfree(vc->vc_decor.theme);
 | |
| 
 | |
|  vc->vc_decor = cfg;
 | |
| 
 | |
|  if (origin == FBCON_DECOR_IO_ORIG_USER)
 | |
|    release_console_sem();
 | |
| 
 | |
|  printk(KERN_INFO "fbcondecor: console %d using theme '%s'\n",
 | |
|       vc->vc_num, vc->vc_decor.theme);
 | |
|  return 0;
 | |
| }
 | |
| 
 | |
| static int fbcon_decor_ioctl_dogetcfg(struct vc_data *vc, struct vc_decor __user *arg)
 | |
| {
 | |
|  struct vc_decor decor;
 | |
|  char __user *tmp;
 | |
| 
 | |
|  if (get_user(tmp, &arg->theme))
 | |
|    return -EFAULT;
 | |
| 
 | |
|  decor = vc->vc_decor;
 | |
|  decor.theme = tmp;
 | |
| 
 | |
|  if (vc->vc_decor.theme) {
 | |
|    if (copy_to_user(tmp, vc->vc_decor.theme, strlen(vc->vc_decor.theme) + 1))
 | |
|      return -EFAULT;
 | |
|  } else
 | |
|    if (put_user(0, tmp))
 | |
|      return -EFAULT;
 | |
| 
 | |
|  if (copy_to_user(arg, &decor, sizeof(struct vc_decor)))
 | |
|    return -EFAULT;
 | |
| 
 | |
|  return 0;
 | |
| }
 | |
| 
 | |
| static int fbcon_decor_ioctl_dosetpic(struct vc_data *vc, struct fb_image __user *arg, unsigned char origin)
 | |
| {
 | |
|  struct fb_image img;
 | |
|  struct fb_info *info;
 | |
|  int len;
 | |
|  u8 *tmp;
 | |
| 
 | |
|  if (vc->vc_num != fg_console)
 | |
|    return -EINVAL;
 | |
| 
 | |
|  info = registered_fb[(int) con2fb_map[vc->vc_num]];
 | |
| 
 | |
|  if (info == NULL)
 | |
|    return -EINVAL;
 | |
| 
 | |
|  if (copy_from_user(&img, arg, sizeof(struct fb_image)))
 | |
|    return -EFAULT;
 | |
| 
 | |
|  if (img.width != info->var.xres || img.height != info->var.yres) {
 | |
|    printk(KERN_ERR "fbcondecor: picture dimensions mismatch\n");
 | |
|    return -EINVAL;
 | |
|  }
 | |
| 
 | |
|  if (img.depth != info->var.bits_per_pixel) {
 | |
|    printk(KERN_ERR "fbcondecor: picture depth mismatch\n");
 | |
|    return -EINVAL;
 | |
|  }
 | |
| 
 | |
|  if (img.depth == 8) {
 | |
|    if (!img.cmap.len || !img.cmap.red || !img.cmap.green ||
 | |
|        !img.cmap.blue)
 | |
|      return -EINVAL;
 | |
| 
 | |
|    tmp = vmalloc(img.cmap.len * 3 * 2);
 | |
|    if (!tmp)
 | |
|      return -ENOMEM;
 | |
| 
 | |
|    if (copy_from_user(tmp, (void __user*)img.cmap.red, img.cmap.len * 2) ||
 | |
|        copy_from_user(tmp + (img.cmap.len << 1),
 | |
|               (void __user*)img.cmap.green, (img.cmap.len << 1)) ||
 | |
|        copy_from_user(tmp + (img.cmap.len << 2),
 | |
|               (void __user*)img.cmap.blue, (img.cmap.len << 1))) {
 | |
|      vfree(tmp);
 | |
|      return -EFAULT;
 | |
|    }
 | |
| 
 | |
|    img.cmap.transp = NULL;
 | |
|    img.cmap.red = (u16*)tmp;
 | |
|    img.cmap.green = img.cmap.red + img.cmap.len;
 | |
|    img.cmap.blue = img.cmap.green + img.cmap.len;
 | |
|  } else {
 | |
|    img.cmap.red = NULL;
 | |
|  }
 | |
| 
 | |
|  len = ((img.depth + 7) >> 3) * img.width * img.height;
 | |
|  tmp = vmalloc(len);
 | |
| 
 | |
|  if (!tmp)
 | |
|    goto out;
 | |
| 
 | |
|  if (copy_from_user(tmp, (void __user*)img.data, len))
 | |
|    goto out;
 | |
| 
 | |
|  img.data = tmp;
 | |
| 
 | |
|  /* If this ioctl is a response to a request from kernel, the console sem
 | |
|   * is already held. */
 | |
|  if (origin == FBCON_DECOR_IO_ORIG_USER)
 | |
|    acquire_console_sem();
 | |
| 
 | |
|  if (info->bgdecor.data)
 | |
|    vfree((u8*)info->bgdecor.data);
 | |
|  if (info->bgdecor.cmap.red)
 | |
|    vfree(info->bgdecor.cmap.red);
 | |
| 
 | |
|  info->bgdecor = img;
 | |
| 
 | |
|  if (origin == FBCON_DECOR_IO_ORIG_USER)
 | |
|    release_console_sem();
 | |
| 
 | |
|  return 0;
 | |
| 
 | |
| out: if (img.cmap.red)
 | |
|    vfree(img.cmap.red);
 | |
|  if (tmp)
 | |
|    vfree(tmp);
 | |
|  return -ENOMEM;
 | |
| }
 | |
| 
 | |
| static int fbcon_decor_ioctl(struct inode * inode, struct file *filp, u_int cmd,
 | |
|      u_long arg)
 | |
| {
 | |
|  struct fbcon_decor_iowrapper __user *wrapper = (void __user*) arg;
 | |
|  struct vc_data *vc = NULL;
 | |
|  unsigned short vc_num = 0;
 | |
|  unsigned char origin = 0;
 | |
|  void __user *data = NULL;
 | |
| 
 | |
|  if (!access_ok(VERIFY_READ, wrapper,
 | |
|      sizeof(struct fbcon_decor_iowrapper)))
 | |
|    return -EFAULT;
 | |
| 
 | |
|  __get_user(vc_num, &wrapper->vc);
 | |
|  __get_user(origin, &wrapper->origin);
 | |
|  __get_user(data, &wrapper->data);
 | |
| 
 | |
|  if (!vc_cons_allocated(vc_num))
 | |
|    return -EINVAL;
 | |
| 
 | |
|  vc = vc_cons[vc_num].d;
 | |
| 
 | |
|  switch (cmd) {
 | |
|  case FBIOCONDECOR_SETPIC:
 | |
|    return fbcon_decor_ioctl_dosetpic(vc, (struct fb_image __user*)data, origin);
 | |
|  case FBIOCONDECOR_SETCFG:
 | |
|    return fbcon_decor_ioctl_dosetcfg(vc, (struct vc_decor*)data, origin);
 | |
|  case FBIOCONDECOR_GETCFG:
 | |
|    return fbcon_decor_ioctl_dogetcfg(vc, (struct vc_decor*)data);
 | |
|  case FBIOCONDECOR_SETSTATE:
 | |
|    return fbcon_decor_ioctl_dosetstate(vc, (unsigned int *)data, origin);
 | |
|  case FBIOCONDECOR_GETSTATE:
 | |
|    return fbcon_decor_ioctl_dogetstate(vc, (unsigned int *)data);
 | |
|  default:
 | |
|    return -ENOIOCTLCMD;
 | |
|  }
 | |
| }
 | |
| 
 | |
| static struct file_operations fbcon_decor_ops = {
 | |
|  .owner = THIS_MODULE,
 | |
|  .ioctl = fbcon_decor_ioctl
 | |
| };
 | |
| 
 | |
| static struct miscdevice fbcon_decor_dev = {
 | |
|  .minor = MISC_DYNAMIC_MINOR,
 | |
|  .name = "fbcondecor",
 | |
|  .fops = &fbcon_decor_ops
 | |
| };
 | |
| 
 | |
| void fbcon_decor_reset(void)
 | |
| {
 | |
|  struct fb_info *info;
 | |
|  struct vc_data *vc;
 | |
|  int i;
 | |
| 
 | |
|  vc = vc_cons[0].d;
 | |
|  info = registered_fb[0];
 | |
| 
 | |
|  for (i = 0; i < num_registered_fb; i++) {
 | |
|    registered_fb[i]->bgdecor.data = NULL;
 | |
|    registered_fb[i]->bgdecor.cmap.red = NULL;
 | |
|  }
 | |
| 
 | |
|  for (i = 0; i < MAX_NR_CONSOLES && vc_cons[i].d; i++) {
 | |
|    vc_cons[i].d->vc_decor.state = vc_cons[i].d->vc_decor.twidth =
 | |
|            vc_cons[i].d->vc_decor.theight = 0;
 | |
|    vc_cons[i].d->vc_decor.theme = NULL;
 | |
|  }
 | |
| 
 | |
|  return;
 | |
| }
 | |
| 
 | |
| int fbcon_decor_init(void)
 | |
| {
 | |
|  int i;
 | |
| 
 | |
|  fbcon_decor_reset();
 | |
| 
 | |
|  if (initialized)
 | |
|    return 0;
 | |
| 
 | |
|  i = misc_register(&fbcon_decor_dev);
 | |
|  if (i) {
 | |
|    printk(KERN_ERR "fbcondecor: failed to register device\n");
 | |
|    return i;
 | |
|  }
 | |
| 
 | |
|  fbcon_decor_call_helper("init", 0);
 | |
|  initialized = 1;
 | |
|  return 0;
 | |
| }
 | |
| 
 | |
| int fbcon_decor_exit(void)
 | |
| {
 | |
|  fbcon_decor_reset();
 | |
|  return 0;
 | |
| }
 | |
| 
 | |
| EXPORT_SYMBOL(fbcon_decor_path);
 |