📄 valkyriefb.c
字号:
#endif if (default_vmode == VMODE_CHOOSE) default_vmode = mac_map_monitor_sense(p->sense); if (!valkyrie_reg_init[default_vmode - 1]) default_vmode = VMODE_640_480_67;#if !defined(CONFIG_MAC) && defined(CONFIG_NVRAM) if (default_cmode == CMODE_NVRAM) default_cmode = nvram_read_byte(NV_CMODE);#endif /* * Reduce the pixel size if we don't have enough VRAM or bandwidth. */ if (default_cmode < CMODE_8 || default_cmode > CMODE_16 || valkyrie_reg_init[default_vmode-1]->pitch[default_cmode] == 0 || valkyrie_vram_reqd(default_vmode, default_cmode) > p->total_vram) default_cmode = CMODE_8; printk(KERN_INFO "using video mode %d and color mode %d.\n", default_vmode, default_cmode);}int __init valkyriefb_init(void){ struct fb_info_valkyrie *p; unsigned long frame_buffer_phys, cmap_regs_phys, flags; int err; char *option = NULL; if (fb_get_options("valkyriefb", &option)) return -ENODEV; valkyriefb_setup(option);#ifdef CONFIG_MAC if (!MACH_IS_MAC) return 0; if (!(mac_bi_data.id == MAC_MODEL_Q630 /* I'm not sure about this one */ || mac_bi_data.id == MAC_MODEL_P588)) return 0; /* Hardcoded addresses... welcome to 68k Macintosh country :-) */ frame_buffer_phys = 0xf9000000; cmap_regs_phys = 0x50f24000; flags = IOMAP_NOCACHE_SER; /* IOMAP_WRITETHROUGH?? */#else /* ppc (!CONFIG_MAC) */ { struct device_node *dp; struct resource r; dp = of_find_node_by_name(NULL, "valkyrie"); if (dp == 0) return 0; if (of_address_to_resource(dp, 0, &r)) { printk(KERN_ERR "can't find address for valkyrie\n"); return 0; } frame_buffer_phys = r.start; cmap_regs_phys = r.start + 0x304000; flags = _PAGE_WRITETHRU; }#endif /* ppc (!CONFIG_MAC) */ p = kzalloc(sizeof(*p), GFP_ATOMIC); if (p == 0) return -ENOMEM; /* Map in frame buffer and registers */ if (!request_mem_region(frame_buffer_phys, 0x100000, "valkyriefb")) { kfree(p); return 0; } p->total_vram = 0x100000; p->frame_buffer_phys = frame_buffer_phys; p->frame_buffer = __ioremap(frame_buffer_phys, p->total_vram, flags); p->cmap_regs_phys = cmap_regs_phys; p->cmap_regs = ioremap(p->cmap_regs_phys, 0x1000); p->valkyrie_regs_phys = cmap_regs_phys+0x6000; p->valkyrie_regs = ioremap(p->valkyrie_regs_phys, 0x1000); err = -ENOMEM; if (p->frame_buffer == NULL || p->cmap_regs == NULL || p->valkyrie_regs == NULL) { printk(KERN_ERR "valkyriefb: couldn't map resources\n"); goto out_free; } valkyrie_choose_mode(p); mac_vmode_to_var(default_vmode, default_cmode, &p->info.var); valkyrie_init_info(&p->info, p); valkyrie_init_fix(&p->info.fix, p); if (valkyriefb_set_par(&p->info)) /* "can't happen" */ printk(KERN_ERR "valkyriefb: can't set default video mode\n"); if ((err = register_framebuffer(&p->info)) != 0) goto out_free; printk(KERN_INFO "fb%d: valkyrie frame buffer device\n", p->info.node); return 0; out_free: if (p->frame_buffer) iounmap(p->frame_buffer); if (p->cmap_regs) iounmap(p->cmap_regs); if (p->valkyrie_regs) iounmap(p->valkyrie_regs); kfree(p); return err;}/* * Get the monitor sense value. */static int read_valkyrie_sense(struct fb_info_valkyrie *p){ int sense, in; out_8(&p->valkyrie_regs->msense.r, 0); /* release all lines */ __delay(20000); sense = ((in = in_8(&p->valkyrie_regs->msense.r)) & 0x70) << 4; /* drive each sense line low in turn and collect the other 2 */ out_8(&p->valkyrie_regs->msense.r, 4); /* drive A low */ __delay(20000); sense |= ((in = in_8(&p->valkyrie_regs->msense.r)) & 0x30); out_8(&p->valkyrie_regs->msense.r, 2); /* drive B low */ __delay(20000); sense |= ((in = in_8(&p->valkyrie_regs->msense.r)) & 0x40) >> 3; sense |= (in & 0x10) >> 2; out_8(&p->valkyrie_regs->msense.r, 1); /* drive C low */ __delay(20000); sense |= ((in = in_8(&p->valkyrie_regs->msense.r)) & 0x60) >> 5; out_8(&p->valkyrie_regs->msense.r, 7); return sense;}/* * This routine takes a user-supplied var, * and picks the best vmode/cmode from it. *//* [bkn] I did a major overhaul of this function. * * Much of the old code was "swiped by jonh from atyfb.c". Because * macmodes has mac_var_to_vmode, I felt that it would be better to * rework this function to use that, instead of reinventing the wheel to * add support for vmode 17. This was reinforced by the fact that * the previously swiped atyfb.c code is no longer there. * * So, I swiped and adapted platinum_var_to_par (from platinumfb.c), replacing * most, but not all, of the old code in the process. One side benefit of * swiping the platinumfb code is that we now have more comprehensible error * messages when a vmode/cmode switch fails. (Most of the error messages are * platinumfb.c, but I added two of my own, and I also changed some commas * into colons to make the messages more consistent with other Linux error * messages.) In addition, I think the new code *might* fix some vmode- * switching oddities, but I'm not sure. * * There may be some more opportunities for cleanup in here, but this is a * good start... */static int valkyrie_var_to_par(struct fb_var_screeninfo *var, struct fb_par_valkyrie *par, const struct fb_info *fb_info){ int vmode, cmode; struct valkyrie_regvals *init; struct fb_info_valkyrie *p = (struct fb_info_valkyrie *) fb_info; if (mac_var_to_vmode(var, &vmode, &cmode) != 0) { printk(KERN_ERR "valkyriefb: can't do %dx%dx%d.\n", var->xres, var->yres, var->bits_per_pixel); return -EINVAL; } /* Check if we know about the wanted video mode */ if (vmode < 1 || vmode > VMODE_MAX || !valkyrie_reg_init[vmode-1]) { printk(KERN_ERR "valkyriefb: vmode %d not valid.\n", vmode); return -EINVAL; } if (cmode != CMODE_8 && cmode != CMODE_16) { printk(KERN_ERR "valkyriefb: cmode %d not valid.\n", cmode); return -EINVAL; } if (var->xres_virtual > var->xres || var->yres_virtual > var->yres || var->xoffset != 0 || var->yoffset != 0) { return -EINVAL; } init = valkyrie_reg_init[vmode-1]; if (init->pitch[cmode] == 0) { printk(KERN_ERR "valkyriefb: vmode %d does not support " "cmode %d.\n", vmode, cmode); return -EINVAL; } if (valkyrie_vram_reqd(vmode, cmode) > p->total_vram) { printk(KERN_ERR "valkyriefb: not enough ram for vmode %d, " "cmode %d.\n", vmode, cmode); return -EINVAL; } par->vmode = vmode; par->cmode = cmode; par->init = init; par->xres = var->xres; par->yres = var->yres; par->vxres = par->xres; par->vyres = par->yres; return 0;}static void valkyrie_init_fix(struct fb_fix_screeninfo *fix, struct fb_info_valkyrie *p){ memset(fix, 0, sizeof(*fix)); strcpy(fix->id, "valkyrie"); fix->mmio_start = p->valkyrie_regs_phys; fix->mmio_len = sizeof(struct valkyrie_regs); fix->type = FB_TYPE_PACKED_PIXELS; fix->smem_start = p->frame_buffer_phys + 0x1000; fix->smem_len = p->total_vram; fix->type_aux = 0; fix->ywrapstep = 0; fix->ypanstep = 0; fix->xpanstep = 0; }/* Fix must already be inited above */static void valkyrie_par_to_fix(struct fb_par_valkyrie *par, struct fb_fix_screeninfo *fix){ fix->smem_len = valkyrie_vram_reqd(par->vmode, par->cmode); fix->visual = (par->cmode == CMODE_8) ? FB_VISUAL_PSEUDOCOLOR : FB_VISUAL_DIRECTCOLOR; fix->line_length = par->vxres << par->cmode; /* ywrapstep, xpanstep, ypanstep */}static void __init valkyrie_init_info(struct fb_info *info, struct fb_info_valkyrie *p){ info->fbops = &valkyriefb_ops; info->screen_base = p->frame_buffer + 0x1000; info->flags = FBINFO_DEFAULT; info->pseudo_palette = p->pseudo_palette; fb_alloc_cmap(&info->cmap, 256, 0); info->par = &p->par;}/* * Parse user speficied options (`video=valkyriefb:') */int __init valkyriefb_setup(char *options){ char *this_opt; if (!options || !*options) return 0; while ((this_opt = strsep(&options, ",")) != NULL) { if (!strncmp(this_opt, "vmode:", 6)) { int vmode = simple_strtoul(this_opt+6, NULL, 0); if (vmode > 0 && vmode <= VMODE_MAX) default_vmode = vmode; } else if (!strncmp(this_opt, "cmode:", 6)) { int depth = simple_strtoul(this_opt+6, NULL, 0); switch (depth) { case 8: default_cmode = CMODE_8; break; case 15: case 16: default_cmode = CMODE_16; break; } } } return 0;}module_init(valkyriefb_init);MODULE_LICENSE("GPL");
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -