Lines Matching +full:int +full:- +full:clock +full:- +full:stable +full:- +full:broken

1 // SPDX-License-Identifier: GPL-2.0-only
9 * Copyright (c) 2001 - 2008 Paul Mundt <lethal@linux-sh.org>
13 * here are some hacked-up formulas:
19 * values, I could just add mode- specific offsets to get the correct mode
22 * left_margin = diwstart_h - borderstart_h;
23 * right_margin = borderstop_h - (diwstart_h + xres);
24 * upper_margin = diwstart_v - borderstart_v;
25 * lower_margin = borderstop_v - (diwstart_h + yres);
27 * hsync_len = borderstart_h + (hsync_total - borderstop_h);
28 * vsync_len = borderstart_v + (vsync_total - borderstop_v);
38 * borderstop_h = borderstart_h + hsync_total - hsync_len;
40 * diwstart_v = borderstart_v - upper_margin;
43 * the benefit of being fully researched, so some modes may be broken.
62 #include <mach-dreamcast/mach/sysasic.h>
81 #define DISP_BASE par->mmio_base
100 /* This is for 60Hz - the VTOTAL is doubled for interlaced modes */
115 struct pvr2_params { unsigned int val; char *name; };
129 unsigned int hsync_total; /* Clocks/line */
130 unsigned int vsync_total; /* Lines/field */
131 unsigned int borderstart_h;
132 unsigned int borderstop_h;
133 unsigned int borderstart_v;
134 unsigned int borderstop_v;
135 unsigned int diwstart_h; /* Horizontal offset of the display field */
136 unsigned int diwstart_v; /* Vertical offset of the display field, for
141 unsigned char is_lowres; /* Is horizontal pixel-doubling enabled? */
168 .height = -1,
169 .width = -1,
173 static int cable_type = CT_VGA;
174 static int video_output = VO_VGA;
176 static int nopan = 0;
177 static int nowrap = 1;
182 static unsigned int do_vmode_full = 0; /* Change the video mode */
183 static unsigned int do_vmode_pan = 0; /* Update the video mode */
186 static unsigned int is_blanked = 0; /* Is the screen blanked? */
193 static unsigned int shdma = PVR2_CASCADE_CHAN;
194 static unsigned int pvr2dma = ONCHIP_NR_DMA_CHANNELS;
200 * PAL-M and PAL-N, but from what I've read both modes parallel PAL and
210 /* XXX: Broken! Don't use... */
226 static int defmode = DEFMODE_NTSC;
229 static inline void pvr2fb_set_pal_type(unsigned int type) in pvr2fb_set_pal_type()
231 struct pvr2fb_par *par = (struct pvr2fb_par *)fb_info->par; in pvr2fb_set_pal_type()
233 fb_writel(type, par->mmio_base + 0x108); in pvr2fb_set_pal_type()
237 unsigned int regno, in pvr2fb_set_pal_entry()
238 unsigned int val) in pvr2fb_set_pal_entry()
240 fb_writel(val, par->mmio_base + 0x1000 + (4 * regno)); in pvr2fb_set_pal_entry()
243 static int pvr2fb_blank(int blank, struct fb_info *info) in pvr2fb_blank()
245 do_blank = blank ? blank : -1; in pvr2fb_blank()
249 static inline unsigned long get_line_length(int xres_virtual, int bpp) in get_line_length()
256 switch (var->bits_per_pixel) { in set_color_bitfields()
259 var->red.offset = 11; var->red.length = 5; in set_color_bitfields()
260 var->green.offset = 5; var->green.length = 6; in set_color_bitfields()
261 var->blue.offset = 0; var->blue.length = 5; in set_color_bitfields()
262 var->transp.offset = 0; var->transp.length = 0; in set_color_bitfields()
265 var->red.offset = 16; var->red.length = 8; in set_color_bitfields()
266 var->green.offset = 8; var->green.length = 8; in set_color_bitfields()
267 var->blue.offset = 0; var->blue.length = 8; in set_color_bitfields()
268 var->transp.offset = 0; var->transp.length = 0; in set_color_bitfields()
272 var->red.offset = 16; var->red.length = 8; in set_color_bitfields()
273 var->green.offset = 8; var->green.length = 8; in set_color_bitfields()
274 var->blue.offset = 0; var->blue.length = 8; in set_color_bitfields()
275 var->transp.offset = 24; var->transp.length = 8; in set_color_bitfields()
280 static int pvr2fb_setcolreg(unsigned int regno, unsigned int red, in pvr2fb_setcolreg()
281 unsigned int green, unsigned int blue, in pvr2fb_setcolreg()
282 unsigned int transp, struct fb_info *info) in pvr2fb_setcolreg()
284 struct pvr2fb_par *par = (struct pvr2fb_par *)info->par; in pvr2fb_setcolreg()
285 unsigned int tmp; in pvr2fb_setcolreg()
287 if (regno > info->cmap.len) in pvr2fb_setcolreg()
295 switch (info->var.bits_per_pixel) { in pvr2fb_setcolreg()
314 pr_debug("Invalid bit depth %d?!?\n", info->var.bits_per_pixel); in pvr2fb_setcolreg()
319 ((u32*)(info->pseudo_palette))[regno] = tmp; in pvr2fb_setcolreg()
333 static int pvr2_init_cable(void) in pvr2_init_cable()
354 static int pvr2fb_set_par(struct fb_info *info) in pvr2fb_set_par()
356 struct pvr2fb_par *par = (struct pvr2fb_par *)info->par; in pvr2fb_set_par()
357 struct fb_var_screeninfo *var = &info->var; in pvr2fb_set_par()
359 unsigned int vtotal; in pvr2fb_set_par()
363 * type in hardware (i.e. switch from VGA<->composite), then change in pvr2fb_set_par()
372 var->vmode &= FB_VMODE_MASK; in pvr2fb_set_par()
373 if (var->vmode & FB_VMODE_INTERLACED && video_output != VO_VGA) in pvr2fb_set_par()
374 par->is_interlaced = 1; in pvr2fb_set_par()
379 if (var->vmode & FB_VMODE_DOUBLE && video_output == VO_VGA) in pvr2fb_set_par()
380 par->is_doublescan = 1; in pvr2fb_set_par()
382 par->hsync_total = var->left_margin + var->xres + var->right_margin + in pvr2fb_set_par()
383 var->hsync_len; in pvr2fb_set_par()
384 par->vsync_total = var->upper_margin + var->yres + var->lower_margin + in pvr2fb_set_par()
385 var->vsync_len; in pvr2fb_set_par()
387 if (var->sync & FB_SYNC_BROADCAST) { in pvr2fb_set_par()
388 vtotal = par->vsync_total; in pvr2fb_set_par()
389 if (par->is_interlaced) in pvr2fb_set_par()
393 /* XXX: Check hardware for PAL-compatibility */ in pvr2fb_set_par()
394 par->borderstart_h = 116; in pvr2fb_set_par()
395 par->borderstart_v = 44; in pvr2fb_set_par()
398 par->borderstart_h = 126; in pvr2fb_set_par()
399 par->borderstart_v = 18; in pvr2fb_set_par()
408 par->borderstart_h = 126; in pvr2fb_set_par()
409 par->borderstart_v = 40; in pvr2fb_set_par()
413 par->diwstart_h = par->borderstart_h + var->left_margin; in pvr2fb_set_par()
414 par->diwstart_v = par->borderstart_v + var->upper_margin; in pvr2fb_set_par()
415 par->borderstop_h = par->diwstart_h + var->xres + in pvr2fb_set_par()
416 var->right_margin; in pvr2fb_set_par()
417 par->borderstop_v = par->diwstart_v + var->yres + in pvr2fb_set_par()
418 var->lower_margin; in pvr2fb_set_par()
420 if (!par->is_interlaced) in pvr2fb_set_par()
421 par->borderstop_v /= 2; in pvr2fb_set_par()
422 if (info->var.xres < 640) in pvr2fb_set_par()
423 par->is_lowres = 1; in pvr2fb_set_par()
425 line_length = get_line_length(var->xres_virtual, var->bits_per_pixel); in pvr2fb_set_par()
426 par->disp_start = info->fix.smem_start + (line_length * var->yoffset) * line_length; in pvr2fb_set_par()
427 info->fix.line_length = line_length; in pvr2fb_set_par()
431 static int pvr2fb_check_var(struct fb_var_screeninfo *var, struct fb_info *info) in pvr2fb_check_var()
433 struct pvr2fb_par *par = (struct pvr2fb_par *)info->par; in pvr2fb_check_var()
434 unsigned int vtotal, hsync_total; in pvr2fb_check_var()
437 if (var->pixclock != TV_CLK && var->pixclock != VGA_CLK) { in pvr2fb_check_var()
438 pr_debug("Invalid pixclock value %d\n", var->pixclock); in pvr2fb_check_var()
439 return -EINVAL; in pvr2fb_check_var()
442 if (var->xres < 320) in pvr2fb_check_var()
443 var->xres = 320; in pvr2fb_check_var()
444 if (var->yres < 240) in pvr2fb_check_var()
445 var->yres = 240; in pvr2fb_check_var()
446 if (var->xres_virtual < var->xres) in pvr2fb_check_var()
447 var->xres_virtual = var->xres; in pvr2fb_check_var()
448 if (var->yres_virtual < var->yres) in pvr2fb_check_var()
449 var->yres_virtual = var->yres; in pvr2fb_check_var()
451 if (var->bits_per_pixel <= 16) in pvr2fb_check_var()
452 var->bits_per_pixel = 16; in pvr2fb_check_var()
453 else if (var->bits_per_pixel <= 24) in pvr2fb_check_var()
454 var->bits_per_pixel = 24; in pvr2fb_check_var()
455 else if (var->bits_per_pixel <= 32) in pvr2fb_check_var()
456 var->bits_per_pixel = 32; in pvr2fb_check_var()
460 if (var->vmode & FB_VMODE_YWRAP) { in pvr2fb_check_var()
461 if (var->xoffset || var->yoffset >= var->yres_virtual) { in pvr2fb_check_var()
462 var->xoffset = var->yoffset = 0; in pvr2fb_check_var()
464 if (var->xoffset > var->xres_virtual - var->xres || in pvr2fb_check_var()
465 var->yoffset > var->yres_virtual - var->yres) in pvr2fb_check_var()
466 var->xoffset = var->yoffset = 0; in pvr2fb_check_var()
469 var->xoffset = var->yoffset = 0; in pvr2fb_check_var()
476 if (var->yres < 480 && video_output == VO_VGA) in pvr2fb_check_var()
477 var->vmode |= FB_VMODE_DOUBLE; in pvr2fb_check_var()
480 var->sync |= FB_SYNC_BROADCAST; in pvr2fb_check_var()
481 var->vmode |= FB_VMODE_INTERLACED; in pvr2fb_check_var()
483 var->sync &= ~FB_SYNC_BROADCAST; in pvr2fb_check_var()
484 var->vmode &= ~FB_VMODE_INTERLACED; in pvr2fb_check_var()
485 var->vmode |= FB_VMODE_NONINTERLACED; in pvr2fb_check_var()
488 if ((var->activate & FB_ACTIVATE_MASK) != FB_ACTIVATE_TEST) { in pvr2fb_check_var()
489 var->right_margin = par->borderstop_h - in pvr2fb_check_var()
490 (par->diwstart_h + var->xres); in pvr2fb_check_var()
491 var->left_margin = par->diwstart_h - par->borderstart_h; in pvr2fb_check_var()
492 var->hsync_len = par->borderstart_h + in pvr2fb_check_var()
493 (par->hsync_total - par->borderstop_h); in pvr2fb_check_var()
495 var->upper_margin = par->diwstart_v - par->borderstart_v; in pvr2fb_check_var()
496 var->lower_margin = par->borderstop_v - in pvr2fb_check_var()
497 (par->diwstart_v + var->yres); in pvr2fb_check_var()
498 var->vsync_len = par->borderstop_v + in pvr2fb_check_var()
499 (par->vsync_total - par->borderstop_v); in pvr2fb_check_var()
502 hsync_total = var->left_margin + var->xres + var->right_margin + in pvr2fb_check_var()
503 var->hsync_len; in pvr2fb_check_var()
504 vtotal = var->upper_margin + var->yres + var->lower_margin + in pvr2fb_check_var()
505 var->vsync_len; in pvr2fb_check_var()
507 if (var->sync & FB_SYNC_BROADCAST) { in pvr2fb_check_var()
508 if (var->vmode & FB_VMODE_INTERLACED) in pvr2fb_check_var()
515 return -EINVAL; in pvr2fb_check_var()
521 return -EINVAL; in pvr2fb_check_var()
527 line_length = get_line_length(var->xres_virtual, var->bits_per_pixel); in pvr2fb_check_var()
528 if (line_length * var->yres_virtual > info->fix.smem_len) in pvr2fb_check_var()
529 return -ENOMEM; in pvr2fb_check_var()
536 struct pvr2fb_par *par = (struct pvr2fb_par *) info->par; in pvr2_update_display()
537 struct fb_var_screeninfo *var = &info->var; in pvr2_update_display()
540 fb_writel(par->disp_start, DISP_DIWADDRL); in pvr2_update_display()
541 fb_writel(par->disp_start + in pvr2_update_display()
542 get_line_length(var->xoffset+var->xres, var->bits_per_pixel), in pvr2_update_display()
548 * very stable. It's probably due to the fact that a lot of the 2D video
554 struct pvr2fb_par *par = (struct pvr2fb_par *) info->par; in pvr2_init_display()
555 struct fb_var_screeninfo *var = &info->var; in pvr2_init_display()
556 unsigned int diw_height, diw_width, diw_modulo = 1; in pvr2_init_display()
557 unsigned int bytesperpixel = var->bits_per_pixel >> 3; in pvr2_init_display()
560 fb_writel((par->vsync_total << 16) | par->hsync_total, DISP_SYNCSIZE); in pvr2_init_display()
565 if (video_output != VO_VGA && par->is_interlaced) in pvr2_init_display()
566 diw_modulo += info->fix.line_length / 4; in pvr2_init_display()
567 diw_height = (par->is_interlaced ? var->yres / 2 : var->yres); in pvr2_init_display()
568 diw_width = get_line_length(var->xres, var->bits_per_pixel) / 4; in pvr2_init_display()
569 fb_writel((diw_modulo << 20) | (--diw_height << 10) | --diw_width, in pvr2_init_display()
573 fb_writel(par->disp_start, DISP_DIWADDRL); in pvr2_init_display()
574 fb_writel(par->disp_start + in pvr2_init_display()
575 get_line_length(var->xoffset+var->xres, var->bits_per_pixel), in pvr2_init_display()
579 fb_writel((par->borderstart_h << 16) | par->borderstop_h, DISP_BRDRHORZ); in pvr2_init_display()
580 fb_writel((par->borderstart_v << 16) | par->borderstop_v, DISP_BRDRVERT); in pvr2_init_display()
584 fb_writel(par->diwstart_h, DISP_DIWHSTRT); in pvr2_init_display()
585 fb_writel((par->diwstart_v << 16) | par->diwstart_v, DISP_DIWVSTRT); in pvr2_init_display()
588 fb_writel((0x16 << 16) | par->is_lowres, DISP_DIWCONF); in pvr2_init_display()
590 /* clock doubler (for VGA), scan doubler, display enable */ in pvr2_init_display()
592 (par->is_doublescan << 1) | 1, DISP_DIWMODE); in pvr2_init_display()
595 fb_writel(fb_readl(DISP_DIWMODE) | (--bytesperpixel << 2), DISP_DIWMODE); in pvr2_init_display()
600 fb_writel(0x100 | ((par->is_interlaced /*|4*/) << 4), DISP_SYNCCONF); in pvr2_init_display()
621 static irqreturn_t __maybe_unused pvr2fb_interrupt(int irq, void *dev_id) in pvr2fb_interrupt()
645 unsigned int nr_pages; in pvr2fb_write()
647 int ret, i; in pvr2fb_write()
649 nr_pages = (count + PAGE_SIZE - 1) >> PAGE_SHIFT; in pvr2fb_write()
653 return -ENOMEM; in pvr2fb_write()
661 * -errno value was returned from GUP. in pvr2fb_write()
667 * Use -EINVAL to represent a mildly desperate guess at in pvr2fb_write()
671 ret = -EINVAL; in pvr2fb_write()
678 dst = (unsigned long)fb_info->screen_base + *ppos; in pvr2fb_write()
683 /* Half-assed contig check */ in pvr2fb_write()
686 if ((*ppos + len) > fb_info->fix.smem_len) { in pvr2fb_write()
687 ret = -ENOSPC; in pvr2fb_write()
698 /* Not contiguous, writeout per-page instead.. */ in pvr2fb_write()
700 if ((*ppos + (i << PAGE_SHIFT)) > fb_info->fix.smem_len) { in pvr2fb_write()
701 ret = -ENOSPC; in pvr2fb_write()
737 static int pvr2_get_param_val(const struct pvr2_params *p, const char *s, in pvr2_get_param_val()
738 int size) in pvr2_get_param_val()
740 int i; in pvr2_get_param_val()
746 return -1; in pvr2_get_param_val()
750 static char *pvr2_get_param_name(const struct pvr2_params *p, int val, in pvr2_get_param_name()
751 int size) in pvr2_get_param_name()
753 int i; in pvr2_get_param_name()
768 * registration. It's expected that the board-specific init code has
772 * default from the modedb. For board-specific modelines, simply define
773 * a per-board modedb.
776 * always going to be VGA for the PCI-based PVR2 boards, but we leave this
777 * in for flexibility anyways. Who knows, maybe someone has tv-out on a
778 * PCI-based version of these things ;-)
780 static int __maybe_unused pvr2fb_common_init(void) in pvr2fb_common_init()
785 fb_info->screen_base = ioremap(pvr2_fix.smem_start, in pvr2fb_common_init()
788 if (!fb_info->screen_base) { in pvr2fb_common_init()
793 par->mmio_base = ioremap(pvr2_fix.mmio_start, in pvr2fb_common_init()
795 if (!par->mmio_base) { in pvr2fb_common_init()
800 fb_memset(fb_info->screen_base, 0, pvr2_fix.smem_len); in pvr2fb_common_init()
805 fb_info->fbops = &pvr2fb_ops; in pvr2fb_common_init()
806 fb_info->fix = pvr2_fix; in pvr2fb_common_init()
807 fb_info->par = currentpar; in pvr2fb_common_init()
808 fb_info->pseudo_palette = currentpar->palette; in pvr2fb_common_init()
809 fb_info->flags = FBINFO_DEFAULT | FBINFO_HWACCEL_YPAN; in pvr2fb_common_init()
817 if (!fb_find_mode(&fb_info->var, fb_info, mode_option, pvr2_modedb, in pvr2fb_common_init()
819 fb_info->var = pvr2_var; in pvr2fb_common_init()
821 fb_alloc_cmap(&fb_info->cmap, 256, 0); in pvr2fb_common_init()
825 /*Must write PIXDEPTH to register before anything is displayed - so force init */ in pvr2fb_common_init()
828 modememused = get_line_length(fb_info->var.xres_virtual, in pvr2fb_common_init()
829 fb_info->var.bits_per_pixel); in pvr2fb_common_init()
830 modememused *= fb_info->var.yres_virtual; in pvr2fb_common_init()
832 rev = fb_readl(par->mmio_base + 0x04); in pvr2fb_common_init()
835 fb_info->fix.id, (rev >> 4) & 0x0f, rev & 0x0f, in pvr2fb_common_init()
837 (unsigned long)(fb_info->fix.smem_len >> 10)); in pvr2fb_common_init()
838 fb_info(fb_info, "Mode %dx%d-%d pitch = %ld cable: %s video output: %s\n", in pvr2fb_common_init()
839 fb_info->var.xres, fb_info->var.yres, in pvr2fb_common_init()
840 fb_info->var.bits_per_pixel, in pvr2fb_common_init()
841 get_line_length(fb_info->var.xres, fb_info->var.bits_per_pixel), in pvr2fb_common_init()
848 pvr2fb_map = sq_remap(fb_info->fix.smem_start, fb_info->fix.smem_len, in pvr2fb_common_init()
849 fb_info->fix.id, PAGE_SHARED); in pvr2fb_common_init()
858 if (fb_info->screen_base) in pvr2fb_common_init()
859 iounmap(fb_info->screen_base); in pvr2fb_common_init()
860 if (par->mmio_base) in pvr2fb_common_init()
861 iounmap(par->mmio_base); in pvr2fb_common_init()
863 return -ENXIO; in pvr2fb_common_init()
867 static int __init pvr2fb_dc_init(void) in pvr2fb_dc_init()
870 return -ENXIO; in pvr2fb_dc_init()
874 fb_info->monspecs.hfmin = 30000; in pvr2fb_dc_init()
875 fb_info->monspecs.hfmax = 70000; in pvr2fb_dc_init()
876 fb_info->monspecs.vfmin = 60; in pvr2fb_dc_init()
877 fb_info->monspecs.vfmax = 60; in pvr2fb_dc_init()
880 fb_info->monspecs.hfmin = 15469; in pvr2fb_dc_init()
881 fb_info->monspecs.hfmax = 15781; in pvr2fb_dc_init()
882 fb_info->monspecs.vfmin = 49; in pvr2fb_dc_init()
883 fb_info->monspecs.vfmax = 51; in pvr2fb_dc_init()
908 return -EBUSY; in pvr2fb_dc_init()
914 return -EBUSY; in pvr2fb_dc_init()
923 if (fb_info->screen_base) { in pvr2fb_dc_exit()
924 iounmap(fb_info->screen_base); in pvr2fb_dc_exit()
925 fb_info->screen_base = NULL; in pvr2fb_dc_exit()
927 if (currentpar->mmio_base) { in pvr2fb_dc_exit()
928 iounmap(currentpar->mmio_base); in pvr2fb_dc_exit()
929 currentpar->mmio_base = NULL; in pvr2fb_dc_exit()
940 static int pvr2fb_pci_probe(struct pci_dev *pdev, in pvr2fb_pci_probe()
943 int ret; in pvr2fb_pci_probe()
966 fb_info->device = &pdev->dev; in pvr2fb_pci_probe()
973 if (fb_info->screen_base) { in pvr2fb_pci_remove()
974 iounmap(fb_info->screen_base); in pvr2fb_pci_remove()
975 fb_info->screen_base = NULL; in pvr2fb_pci_remove()
977 if (currentpar->mmio_base) { in pvr2fb_pci_remove()
978 iounmap(currentpar->mmio_base); in pvr2fb_pci_remove()
979 currentpar->mmio_base = NULL; in pvr2fb_pci_remove()
1000 static int __init pvr2fb_pci_init(void) in pvr2fb_pci_init()
1017 * <xres>x<yres>[-<bpp>][@<refresh>] or,
1018 * <name>[-<bpp>][@<refresh>] Startup using this video mode
1022 static int __init pvr2fb_setup(char *options) in pvr2fb_setup()
1061 int (*init)(void);
1074 static int __init pvr2fb_init(void) in pvr2fb_init()
1076 int i, ret = -ENODEV; in pvr2fb_init()
1082 return -ENODEV; in pvr2fb_init()
1088 return -ENOMEM; in pvr2fb_init()
1090 currentpar = fb_info->par; in pvr2fb_init()
1095 if (!pvr_board->init) in pvr2fb_init()
1098 ret = pvr_board->init(); in pvr2fb_init()
1102 pvr_board->name); in pvr2fb_init()
1113 int i; in pvr2fb_exit()
1118 if (pvr_board->exit) in pvr2fb_exit()
1119 pvr_board->exit(); in pvr2fb_exit()
1133 MODULE_AUTHOR("Paul Mundt <lethal@linux-sh.org>, M. R. Brown <mrbrown@0xd6.org>");