diff options
Diffstat (limited to 'drivers/video/cirrusfb.c')
-rw-r--r-- | drivers/video/cirrusfb.c | 192 |
1 files changed, 83 insertions, 109 deletions
diff --git a/drivers/video/cirrusfb.c b/drivers/video/cirrusfb.c index 4e34a6817435..573bd456e5ab 100644 --- a/drivers/video/cirrusfb.c +++ b/drivers/video/cirrusfb.c | |||
@@ -334,22 +334,6 @@ struct cirrusfb_regs { | |||
334 | long multiplexing; | 334 | long multiplexing; |
335 | long mclk; | 335 | long mclk; |
336 | long divMCLK; | 336 | long divMCLK; |
337 | |||
338 | long HorizRes; /* The x resolution in pixel */ | ||
339 | long HorizTotal; | ||
340 | long HorizDispEnd; | ||
341 | long HorizBlankStart; | ||
342 | long HorizBlankEnd; | ||
343 | long HorizSyncStart; | ||
344 | long HorizSyncEnd; | ||
345 | |||
346 | long VertRes; /* the physical y resolution in scanlines */ | ||
347 | long VertTotal; | ||
348 | long VertDispEnd; | ||
349 | long VertSyncStart; | ||
350 | long VertSyncEnd; | ||
351 | long VertBlankStart; | ||
352 | long VertBlankEnd; | ||
353 | }; | 337 | }; |
354 | 338 | ||
355 | #ifdef CIRRUSFB_DEBUG | 339 | #ifdef CIRRUSFB_DEBUG |
@@ -664,8 +648,6 @@ static int cirrusfb_decode_var(const struct fb_var_screeninfo *var, | |||
664 | long maxclock; | 648 | long maxclock; |
665 | int maxclockidx = var->bits_per_pixel >> 3; | 649 | int maxclockidx = var->bits_per_pixel >> 3; |
666 | struct cirrusfb_info *cinfo = info->par; | 650 | struct cirrusfb_info *cinfo = info->par; |
667 | int xres, hfront, hsync, hback; | ||
668 | int yres, vfront, vsync, vback; | ||
669 | 651 | ||
670 | switch (var->bits_per_pixel) { | 652 | switch (var->bits_per_pixel) { |
671 | case 1: | 653 | case 1: |
@@ -723,7 +705,7 @@ static int cirrusfb_decode_var(const struct fb_var_screeninfo *var, | |||
723 | switch (var->bits_per_pixel) { | 705 | switch (var->bits_per_pixel) { |
724 | case 16: | 706 | case 16: |
725 | case 32: | 707 | case 32: |
726 | if (regs->HorizRes <= 800) | 708 | if (var->xres <= 800) |
727 | /* Xbh has this type of clock for 32-bit */ | 709 | /* Xbh has this type of clock for 32-bit */ |
728 | freq /= 2; | 710 | freq /= 2; |
729 | break; | 711 | break; |
@@ -735,57 +717,6 @@ static int cirrusfb_decode_var(const struct fb_var_screeninfo *var, | |||
735 | regs->mclk = cirrusfb_get_mclk(freq, var->bits_per_pixel, | 717 | regs->mclk = cirrusfb_get_mclk(freq, var->bits_per_pixel, |
736 | ®s->divMCLK); | 718 | ®s->divMCLK); |
737 | 719 | ||
738 | xres = var->xres; | ||
739 | hfront = var->right_margin; | ||
740 | hsync = var->hsync_len; | ||
741 | hback = var->left_margin; | ||
742 | |||
743 | yres = var->yres; | ||
744 | vfront = var->lower_margin; | ||
745 | vsync = var->vsync_len; | ||
746 | vback = var->upper_margin; | ||
747 | |||
748 | if (var->vmode & FB_VMODE_DOUBLE) { | ||
749 | yres *= 2; | ||
750 | vfront *= 2; | ||
751 | vsync *= 2; | ||
752 | vback *= 2; | ||
753 | } else if (var->vmode & FB_VMODE_INTERLACED) { | ||
754 | yres = (yres + 1) / 2; | ||
755 | vfront = (vfront + 1) / 2; | ||
756 | vsync = (vsync + 1) / 2; | ||
757 | vback = (vback + 1) / 2; | ||
758 | } | ||
759 | regs->HorizRes = xres; | ||
760 | regs->HorizTotal = (xres + hfront + hsync + hback) / 8 - 5; | ||
761 | regs->HorizDispEnd = xres / 8 - 1; | ||
762 | regs->HorizBlankStart = xres / 8; | ||
763 | /* does not count with "-5" */ | ||
764 | regs->HorizBlankEnd = regs->HorizTotal + 5; | ||
765 | regs->HorizSyncStart = (xres + hfront) / 8 + 1; | ||
766 | regs->HorizSyncEnd = (xres + hfront + hsync) / 8 + 1; | ||
767 | |||
768 | regs->VertRes = yres; | ||
769 | regs->VertTotal = yres + vfront + vsync + vback - 2; | ||
770 | regs->VertDispEnd = yres - 1; | ||
771 | regs->VertBlankStart = yres; | ||
772 | regs->VertBlankEnd = regs->VertTotal; | ||
773 | regs->VertSyncStart = yres + vfront - 1; | ||
774 | regs->VertSyncEnd = yres + vfront + vsync - 1; | ||
775 | |||
776 | if (regs->VertRes >= 1024) { | ||
777 | regs->VertTotal /= 2; | ||
778 | regs->VertSyncStart /= 2; | ||
779 | regs->VertSyncEnd /= 2; | ||
780 | regs->VertDispEnd /= 2; | ||
781 | } | ||
782 | if (regs->multiplexing) { | ||
783 | regs->HorizTotal /= 2; | ||
784 | regs->HorizSyncStart /= 2; | ||
785 | regs->HorizSyncEnd /= 2; | ||
786 | regs->HorizDispEnd /= 2; | ||
787 | } | ||
788 | |||
789 | return 0; | 720 | return 0; |
790 | } | 721 | } |
791 | 722 | ||
@@ -823,6 +754,8 @@ static int cirrusfb_set_par_foo(struct fb_info *info) | |||
823 | unsigned char tmp; | 754 | unsigned char tmp; |
824 | int offset = 0, err; | 755 | int offset = 0, err; |
825 | const struct cirrusfb_board_info_rec *bi; | 756 | const struct cirrusfb_board_info_rec *bi; |
757 | int hdispend, hsyncstart, hsyncend, htotal; | ||
758 | int yres, vdispend, vsyncstart, vsyncend, vtotal; | ||
826 | 759 | ||
827 | DPRINTK("ENTER\n"); | 760 | DPRINTK("ENTER\n"); |
828 | DPRINTK("Requested mode: %dx%dx%d\n", | 761 | DPRINTK("Requested mode: %dx%dx%d\n", |
@@ -840,76 +773,117 @@ static int cirrusfb_set_par_foo(struct fb_info *info) | |||
840 | 773 | ||
841 | bi = &cirrusfb_board_info[cinfo->btype]; | 774 | bi = &cirrusfb_board_info[cinfo->btype]; |
842 | 775 | ||
776 | hsyncstart = var->xres + var->right_margin; | ||
777 | hsyncend = hsyncstart + var->hsync_len; | ||
778 | htotal = (hsyncend + var->left_margin) / 8 - 5; | ||
779 | hdispend = var->xres / 8 - 1; | ||
780 | hsyncstart = hsyncstart / 8 + 1; | ||
781 | hsyncend = hsyncend / 8 + 1; | ||
782 | |||
783 | yres = var->yres; | ||
784 | vsyncstart = yres + var->lower_margin; | ||
785 | vsyncend = vsyncstart + var->vsync_len; | ||
786 | vtotal = vsyncend + var->upper_margin; | ||
787 | vdispend = yres - 1; | ||
788 | |||
789 | if (var->vmode & FB_VMODE_DOUBLE) { | ||
790 | yres *= 2; | ||
791 | vsyncstart *= 2; | ||
792 | vsyncend *= 2; | ||
793 | vtotal *= 2; | ||
794 | } else if (var->vmode & FB_VMODE_INTERLACED) { | ||
795 | yres = (yres + 1) / 2; | ||
796 | vsyncstart = (vsyncstart + 1) / 2; | ||
797 | vsyncend = (vsyncend + 1) / 2; | ||
798 | vtotal = (vtotal + 1) / 2; | ||
799 | } | ||
800 | |||
801 | vtotal -= 2; | ||
802 | vsyncstart -= 1; | ||
803 | vsyncend -= 1; | ||
804 | |||
805 | if (yres >= 1024) { | ||
806 | vtotal /= 2; | ||
807 | vsyncstart /= 2; | ||
808 | vsyncend /= 2; | ||
809 | vdispend /= 2; | ||
810 | } | ||
811 | if (regs.multiplexing) { | ||
812 | htotal /= 2; | ||
813 | hsyncstart /= 2; | ||
814 | hsyncend /= 2; | ||
815 | hdispend /= 2; | ||
816 | } | ||
843 | /* unlock register VGA_CRTC_H_TOTAL..CRT7 */ | 817 | /* unlock register VGA_CRTC_H_TOTAL..CRT7 */ |
844 | vga_wcrt(regbase, VGA_CRTC_V_SYNC_END, 0x20); /* previously: 0x00) */ | 818 | vga_wcrt(regbase, VGA_CRTC_V_SYNC_END, 0x20); /* previously: 0x00) */ |
845 | 819 | ||
846 | /* if debugging is enabled, all parameters get output before writing */ | 820 | /* if debugging is enabled, all parameters get output before writing */ |
847 | DPRINTK("CRT0: %ld\n", regs.HorizTotal); | 821 | DPRINTK("CRT0: %d\n", htotal); |
848 | vga_wcrt(regbase, VGA_CRTC_H_TOTAL, regs.HorizTotal); | 822 | vga_wcrt(regbase, VGA_CRTC_H_TOTAL, htotal); |
849 | 823 | ||
850 | DPRINTK("CRT1: %ld\n", regs.HorizDispEnd); | 824 | DPRINTK("CRT1: %d\n", hdispend); |
851 | vga_wcrt(regbase, VGA_CRTC_H_DISP, regs.HorizDispEnd); | 825 | vga_wcrt(regbase, VGA_CRTC_H_DISP, hdispend); |
852 | 826 | ||
853 | DPRINTK("CRT2: %ld\n", regs.HorizBlankStart); | 827 | DPRINTK("CRT2: %d\n", var->xres / 8); |
854 | vga_wcrt(regbase, VGA_CRTC_H_BLANK_START, regs.HorizBlankStart); | 828 | vga_wcrt(regbase, VGA_CRTC_H_BLANK_START, var->xres / 8); |
855 | 829 | ||
856 | /* + 128: Compatible read */ | 830 | /* + 128: Compatible read */ |
857 | DPRINTK("CRT3: 128+%ld\n", regs.HorizBlankEnd % 32); | 831 | DPRINTK("CRT3: 128+%d\n", (htotal + 5) % 32); |
858 | vga_wcrt(regbase, VGA_CRTC_H_BLANK_END, | 832 | vga_wcrt(regbase, VGA_CRTC_H_BLANK_END, |
859 | 128 + (regs.HorizBlankEnd % 32)); | 833 | 128 + ((htotal + 5) % 32)); |
860 | 834 | ||
861 | DPRINTK("CRT4: %ld\n", regs.HorizSyncStart); | 835 | DPRINTK("CRT4: %d\n", hsyncstart); |
862 | vga_wcrt(regbase, VGA_CRTC_H_SYNC_START, regs.HorizSyncStart); | 836 | vga_wcrt(regbase, VGA_CRTC_H_SYNC_START, hsyncstart); |
863 | 837 | ||
864 | tmp = regs.HorizSyncEnd % 32; | 838 | tmp = hsyncend % 32; |
865 | if (regs.HorizBlankEnd & 32) | 839 | if ((htotal + 5) & 32) |
866 | tmp += 128; | 840 | tmp += 128; |
867 | DPRINTK("CRT5: %d\n", tmp); | 841 | DPRINTK("CRT5: %d\n", tmp); |
868 | vga_wcrt(regbase, VGA_CRTC_H_SYNC_END, tmp); | 842 | vga_wcrt(regbase, VGA_CRTC_H_SYNC_END, tmp); |
869 | 843 | ||
870 | DPRINTK("CRT6: %ld\n", regs.VertTotal & 0xff); | 844 | DPRINTK("CRT6: %d\n", vtotal & 0xff); |
871 | vga_wcrt(regbase, VGA_CRTC_V_TOTAL, (regs.VertTotal & 0xff)); | 845 | vga_wcrt(regbase, VGA_CRTC_V_TOTAL, vtotal & 0xff); |
872 | 846 | ||
873 | tmp = 16; /* LineCompare bit #9 */ | 847 | tmp = 16; /* LineCompare bit #9 */ |
874 | if (regs.VertTotal & 256) | 848 | if (vtotal & 256) |
875 | tmp |= 1; | 849 | tmp |= 1; |
876 | if (regs.VertDispEnd & 256) | 850 | if (vdispend & 256) |
877 | tmp |= 2; | 851 | tmp |= 2; |
878 | if (regs.VertSyncStart & 256) | 852 | if (vsyncstart & 256) |
879 | tmp |= 4; | 853 | tmp |= 4; |
880 | if (regs.VertBlankStart & 256) | 854 | if ((vdispend + 1) & 256) |
881 | tmp |= 8; | 855 | tmp |= 8; |
882 | if (regs.VertTotal & 512) | 856 | if (vtotal & 512) |
883 | tmp |= 32; | 857 | tmp |= 32; |
884 | if (regs.VertDispEnd & 512) | 858 | if (vdispend & 512) |
885 | tmp |= 64; | 859 | tmp |= 64; |
886 | if (regs.VertSyncStart & 512) | 860 | if (vsyncstart & 512) |
887 | tmp |= 128; | 861 | tmp |= 128; |
888 | DPRINTK("CRT7: %d\n", tmp); | 862 | DPRINTK("CRT7: %d\n", tmp); |
889 | vga_wcrt(regbase, VGA_CRTC_OVERFLOW, tmp); | 863 | vga_wcrt(regbase, VGA_CRTC_OVERFLOW, tmp); |
890 | 864 | ||
891 | tmp = 0x40; /* LineCompare bit #8 */ | 865 | tmp = 0x40; /* LineCompare bit #8 */ |
892 | if (regs.VertBlankStart & 512) | 866 | if ((vdispend + 1) & 512) |
893 | tmp |= 0x20; | 867 | tmp |= 0x20; |
894 | if (var->vmode & FB_VMODE_DOUBLE) | 868 | if (var->vmode & FB_VMODE_DOUBLE) |
895 | tmp |= 0x80; | 869 | tmp |= 0x80; |
896 | DPRINTK("CRT9: %d\n", tmp); | 870 | DPRINTK("CRT9: %d\n", tmp); |
897 | vga_wcrt(regbase, VGA_CRTC_MAX_SCAN, tmp); | 871 | vga_wcrt(regbase, VGA_CRTC_MAX_SCAN, tmp); |
898 | 872 | ||
899 | DPRINTK("CRT10: %ld\n", regs.VertSyncStart & 0xff); | 873 | DPRINTK("CRT10: %d\n", vsyncstart & 0xff); |
900 | vga_wcrt(regbase, VGA_CRTC_V_SYNC_START, regs.VertSyncStart & 0xff); | 874 | vga_wcrt(regbase, VGA_CRTC_V_SYNC_START, vsyncstart & 0xff); |
901 | 875 | ||
902 | DPRINTK("CRT11: 64+32+%ld\n", regs.VertSyncEnd % 16); | 876 | DPRINTK("CRT11: 64+32+%d\n", vsyncend % 16); |
903 | vga_wcrt(regbase, VGA_CRTC_V_SYNC_END, regs.VertSyncEnd % 16 + 64 + 32); | 877 | vga_wcrt(regbase, VGA_CRTC_V_SYNC_END, vsyncend % 16 + 64 + 32); |
904 | 878 | ||
905 | DPRINTK("CRT12: %ld\n", regs.VertDispEnd & 0xff); | 879 | DPRINTK("CRT12: %d\n", vdispend & 0xff); |
906 | vga_wcrt(regbase, VGA_CRTC_V_DISP_END, regs.VertDispEnd & 0xff); | 880 | vga_wcrt(regbase, VGA_CRTC_V_DISP_END, vdispend & 0xff); |
907 | 881 | ||
908 | DPRINTK("CRT15: %ld\n", regs.VertBlankStart & 0xff); | 882 | DPRINTK("CRT15: %d\n", (vdispend + 1) & 0xff); |
909 | vga_wcrt(regbase, VGA_CRTC_V_BLANK_START, regs.VertBlankStart & 0xff); | 883 | vga_wcrt(regbase, VGA_CRTC_V_BLANK_START, (vdispend + 1) & 0xff); |
910 | 884 | ||
911 | DPRINTK("CRT16: %ld\n", regs.VertBlankEnd & 0xff); | 885 | DPRINTK("CRT16: %d\n", vtotal & 0xff); |
912 | vga_wcrt(regbase, VGA_CRTC_V_BLANK_END, regs.VertBlankEnd & 0xff); | 886 | vga_wcrt(regbase, VGA_CRTC_V_BLANK_END, vtotal & 0xff); |
913 | 887 | ||
914 | DPRINTK("CRT18: 0xff\n"); | 888 | DPRINTK("CRT18: 0xff\n"); |
915 | vga_wcrt(regbase, VGA_CRTC_LINE_COMPARE, 0xff); | 889 | vga_wcrt(regbase, VGA_CRTC_LINE_COMPARE, 0xff); |
@@ -917,13 +891,13 @@ static int cirrusfb_set_par_foo(struct fb_info *info) | |||
917 | tmp = 0; | 891 | tmp = 0; |
918 | if (var->vmode & FB_VMODE_INTERLACED) | 892 | if (var->vmode & FB_VMODE_INTERLACED) |
919 | tmp |= 1; | 893 | tmp |= 1; |
920 | if (regs.HorizBlankEnd & 64) | 894 | if ((htotal + 5) & 64) |
921 | tmp |= 16; | 895 | tmp |= 16; |
922 | if (regs.HorizBlankEnd & 128) | 896 | if ((htotal + 5) & 128) |
923 | tmp |= 32; | 897 | tmp |= 32; |
924 | if (regs.VertBlankEnd & 256) | 898 | if (vtotal & 256) |
925 | tmp |= 64; | 899 | tmp |= 64; |
926 | if (regs.VertBlankEnd & 512) | 900 | if (vtotal & 512) |
927 | tmp |= 128; | 901 | tmp |= 128; |
928 | 902 | ||
929 | DPRINTK("CRT1a: %d\n", tmp); | 903 | DPRINTK("CRT1a: %d\n", tmp); |
@@ -948,7 +922,7 @@ static int cirrusfb_set_par_foo(struct fb_info *info) | |||
948 | DPRINTK("CL_SEQR1B: %ld\n", (long) tmp); | 922 | DPRINTK("CL_SEQR1B: %ld\n", (long) tmp); |
949 | vga_wseq(regbase, CL_SEQR1B, tmp); | 923 | vga_wseq(regbase, CL_SEQR1B, tmp); |
950 | 924 | ||
951 | if (regs.VertRes >= 1024) | 925 | if (yres >= 1024) |
952 | /* 1280x1024 */ | 926 | /* 1280x1024 */ |
953 | vga_wcrt(regbase, VGA_CRTC_MODE, 0xc7); | 927 | vga_wcrt(regbase, VGA_CRTC_MODE, 0xc7); |
954 | else | 928 | else |
@@ -962,7 +936,7 @@ static int cirrusfb_set_par_foo(struct fb_info *info) | |||
962 | /* don't know if it would hurt to also program this if no interlaced */ | 936 | /* don't know if it would hurt to also program this if no interlaced */ |
963 | /* mode is used, but I feel better this way.. :-) */ | 937 | /* mode is used, but I feel better this way.. :-) */ |
964 | if (var->vmode & FB_VMODE_INTERLACED) | 938 | if (var->vmode & FB_VMODE_INTERLACED) |
965 | vga_wcrt(regbase, VGA_CRTC_REGS, regs.HorizTotal / 2); | 939 | vga_wcrt(regbase, VGA_CRTC_REGS, htotal / 2); |
966 | else | 940 | else |
967 | vga_wcrt(regbase, VGA_CRTC_REGS, 0x00); /* interlace control */ | 941 | vga_wcrt(regbase, VGA_CRTC_REGS, 0x00); /* interlace control */ |
968 | 942 | ||
@@ -1208,7 +1182,7 @@ static int cirrusfb_set_par_foo(struct fb_info *info) | |||
1208 | 1182 | ||
1209 | case BT_ALPINE: | 1183 | case BT_ALPINE: |
1210 | DPRINTK(" (for GD543x)\n"); | 1184 | DPRINTK(" (for GD543x)\n"); |
1211 | if (regs.HorizRes >= 1024) | 1185 | if (var->xres >= 1024) |
1212 | vga_wseq(regbase, CL_SEQR7, 0xa7); | 1186 | vga_wseq(regbase, CL_SEQR7, 0xa7); |
1213 | else | 1187 | else |
1214 | vga_wseq(regbase, CL_SEQR7, 0xa3); | 1188 | vga_wseq(regbase, CL_SEQR7, 0xa3); |