aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/video/intelfb/intelfbhw.c
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/video/intelfb/intelfbhw.c')
-rw-r--r--drivers/video/intelfb/intelfbhw.c149
1 files changed, 93 insertions, 56 deletions
diff --git a/drivers/video/intelfb/intelfbhw.c b/drivers/video/intelfb/intelfbhw.c
index a3a94642b79b..bf742aad08e9 100644
--- a/drivers/video/intelfb/intelfbhw.c
+++ b/drivers/video/intelfb/intelfbhw.c
@@ -61,67 +61,71 @@ struct pll_min_max plls[PLLS_MAX] = {
61}; 61};
62 62
63int 63int
64intelfbhw_get_chipset(struct pci_dev *pdev, const char **name, int *chipset, 64intelfbhw_get_chipset(struct pci_dev *pdev, struct intelfb_info *dinfo)
65 int *mobile)
66{ 65{
67 u32 tmp; 66 u32 tmp;
68 67 if (!pdev || !dinfo)
69 if (!pdev || !name || !chipset || !mobile)
70 return 1; 68 return 1;
71 69
72 switch (pdev->device) { 70 switch (pdev->device) {
73 case PCI_DEVICE_ID_INTEL_830M: 71 case PCI_DEVICE_ID_INTEL_830M:
74 *name = "Intel(R) 830M"; 72 dinfo->name = "Intel(R) 830M";
75 *chipset = INTEL_830M; 73 dinfo->chipset = INTEL_830M;
76 *mobile = 1; 74 dinfo->mobile = 1;
75 dinfo->pll_index = PLLS_I8xx;
77 return 0; 76 return 0;
78 case PCI_DEVICE_ID_INTEL_845G: 77 case PCI_DEVICE_ID_INTEL_845G:
79 *name = "Intel(R) 845G"; 78 dinfo->name = "Intel(R) 845G";
80 *chipset = INTEL_845G; 79 dinfo->chipset = INTEL_845G;
81 *mobile = 0; 80 dinfo->mobile = 0;
81 dinfo->pll_index = PLLS_I8xx;
82 return 0; 82 return 0;
83 case PCI_DEVICE_ID_INTEL_85XGM: 83 case PCI_DEVICE_ID_INTEL_85XGM:
84 tmp = 0; 84 tmp = 0;
85 *mobile = 1; 85 dinfo->mobile = 1;
86 dinfo->pll_index = PLLS_I8xx;
86 pci_read_config_dword(pdev, INTEL_85X_CAPID, &tmp); 87 pci_read_config_dword(pdev, INTEL_85X_CAPID, &tmp);
87 switch ((tmp >> INTEL_85X_VARIANT_SHIFT) & 88 switch ((tmp >> INTEL_85X_VARIANT_SHIFT) &
88 INTEL_85X_VARIANT_MASK) { 89 INTEL_85X_VARIANT_MASK) {
89 case INTEL_VAR_855GME: 90 case INTEL_VAR_855GME:
90 *name = "Intel(R) 855GME"; 91 dinfo->name = "Intel(R) 855GME";
91 *chipset = INTEL_855GME; 92 dinfo->chipset = INTEL_855GME;
92 return 0; 93 return 0;
93 case INTEL_VAR_855GM: 94 case INTEL_VAR_855GM:
94 *name = "Intel(R) 855GM"; 95 dinfo->name = "Intel(R) 855GM";
95 *chipset = INTEL_855GM; 96 dinfo->chipset = INTEL_855GM;
96 return 0; 97 return 0;
97 case INTEL_VAR_852GME: 98 case INTEL_VAR_852GME:
98 *name = "Intel(R) 852GME"; 99 dinfo->name = "Intel(R) 852GME";
99 *chipset = INTEL_852GME; 100 dinfo->chipset = INTEL_852GME;
100 return 0; 101 return 0;
101 case INTEL_VAR_852GM: 102 case INTEL_VAR_852GM:
102 *name = "Intel(R) 852GM"; 103 dinfo->name = "Intel(R) 852GM";
103 *chipset = INTEL_852GM; 104 dinfo->chipset = INTEL_852GM;
104 return 0; 105 return 0;
105 default: 106 default:
106 *name = "Intel(R) 852GM/855GM"; 107 dinfo->name = "Intel(R) 852GM/855GM";
107 *chipset = INTEL_85XGM; 108 dinfo->chipset = INTEL_85XGM;
108 return 0; 109 return 0;
109 } 110 }
110 break; 111 break;
111 case PCI_DEVICE_ID_INTEL_865G: 112 case PCI_DEVICE_ID_INTEL_865G:
112 *name = "Intel(R) 865G"; 113 dinfo->name = "Intel(R) 865G";
113 *chipset = INTEL_865G; 114 dinfo->chipset = INTEL_865G;
114 *mobile = 0; 115 dinfo->mobile = 0;
116 dinfo->pll_index = PLLS_I8xx;
115 return 0; 117 return 0;
116 case PCI_DEVICE_ID_INTEL_915G: 118 case PCI_DEVICE_ID_INTEL_915G:
117 *name = "Intel(R) 915G"; 119 dinfo->name = "Intel(R) 915G";
118 *chipset = INTEL_915G; 120 dinfo->chipset = INTEL_915G;
119 *mobile = 0; 121 dinfo->mobile = 0;
122 dinfo->pll_index = PLLS_I9xx;
120 return 0; 123 return 0;
121 case PCI_DEVICE_ID_INTEL_915GM: 124 case PCI_DEVICE_ID_INTEL_915GM:
122 *name = "Intel(R) 915GM"; 125 dinfo->name = "Intel(R) 915GM";
123 *chipset = INTEL_915GM; 126 dinfo->chipset = INTEL_915GM;
124 *mobile = 1; 127 dinfo->mobile = 1;
128 dinfo->pll_index = PLLS_I9xx;
125 return 0; 129 return 0;
126 default: 130 default:
127 return 1; 131 return 1;
@@ -549,14 +553,33 @@ intelfbhw_read_hw_state(struct intelfb_info *dinfo, struct intelfb_hwstate *hw,
549} 553}
550 554
551 555
556static int calc_vclock3(int index, int m, int n, int p)
557{
558 return PLL_REFCLK * m / n / p;
559}
560
561static int calc_vclock(int index, int m1, int m2, int n, int p1, int p2)
562{
563 switch(index)
564 {
565 case PLLS_I9xx:
566 return ((PLL_REFCLK * (5 * (m1 + 2) + (m2 + 2)) / (n + 2) /
567 ((p1)) * (p2 ? 10 : 5)));
568 case PLLS_I8xx:
569 default:
570 return ((PLL_REFCLK * (5 * (m1 + 2) + (m2 + 2)) / (n + 2) /
571 ((p1+2) * (1 << (p2 + 1)))));
572 }
573}
574
552void 575void
553intelfbhw_print_hw_state(struct intelfb_info *dinfo, struct intelfb_hwstate *hw) 576intelfbhw_print_hw_state(struct intelfb_info *dinfo, struct intelfb_hwstate *hw)
554{ 577{
555#if REGDUMP 578#if REGDUMP
556 int i, m1, m2, n, p1, p2; 579 int i, m1, m2, n, p1, p2;
557 580 int index = dinfo->pll_index;
558 DBG_MSG("intelfbhw_print_hw_state\n"); 581 DBG_MSG("intelfbhw_print_hw_state\n");
559 582
560 if (!hw || !dinfo) 583 if (!hw || !dinfo)
561 return; 584 return;
562 /* Read in as much of the HW state as possible. */ 585 /* Read in as much of the HW state as possible. */
@@ -573,9 +596,10 @@ intelfbhw_print_hw_state(struct intelfb_info *dinfo, struct intelfb_hwstate *hw)
573 p1 = (hw->vga_pd >> VGAPD_0_P1_SHIFT) & DPLL_P1_MASK; 596 p1 = (hw->vga_pd >> VGAPD_0_P1_SHIFT) & DPLL_P1_MASK;
574 p2 = (hw->vga_pd >> VGAPD_0_P2_SHIFT) & DPLL_P2_MASK; 597 p2 = (hw->vga_pd >> VGAPD_0_P2_SHIFT) & DPLL_P2_MASK;
575 printk(" VGA0: (m1, m2, n, p1, p2) = (%d, %d, %d, %d, %d)\n", 598 printk(" VGA0: (m1, m2, n, p1, p2) = (%d, %d, %d, %d, %d)\n",
576 m1, m2, n, p1, p2); 599 m1, m2, n, p1, p2);
577 printk(" VGA0: clock is %d\n", CALC_VCLOCK(m1, m2, n, p1, p2)); 600 printk(" VGA0: clock is %d\n",
578 601 calc_vclock(index, m1, m2, n, p1, p2));
602
579 n = (hw->vga1_divisor >> FP_N_DIVISOR_SHIFT) & FP_DIVISOR_MASK; 603 n = (hw->vga1_divisor >> FP_N_DIVISOR_SHIFT) & FP_DIVISOR_MASK;
580 m1 = (hw->vga1_divisor >> FP_M1_DIVISOR_SHIFT) & FP_DIVISOR_MASK; 604 m1 = (hw->vga1_divisor >> FP_M1_DIVISOR_SHIFT) & FP_DIVISOR_MASK;
581 m2 = (hw->vga1_divisor >> FP_M2_DIVISOR_SHIFT) & FP_DIVISOR_MASK; 605 m2 = (hw->vga1_divisor >> FP_M2_DIVISOR_SHIFT) & FP_DIVISOR_MASK;
@@ -585,16 +609,16 @@ intelfbhw_print_hw_state(struct intelfb_info *dinfo, struct intelfb_hwstate *hw)
585 p1 = (hw->vga_pd >> VGAPD_1_P1_SHIFT) & DPLL_P1_MASK; 609 p1 = (hw->vga_pd >> VGAPD_1_P1_SHIFT) & DPLL_P1_MASK;
586 p2 = (hw->vga_pd >> VGAPD_1_P2_SHIFT) & DPLL_P2_MASK; 610 p2 = (hw->vga_pd >> VGAPD_1_P2_SHIFT) & DPLL_P2_MASK;
587 printk(" VGA1: (m1, m2, n, p1, p2) = (%d, %d, %d, %d, %d)\n", 611 printk(" VGA1: (m1, m2, n, p1, p2) = (%d, %d, %d, %d, %d)\n",
588 m1, m2, n, p1, p2); 612 m1, m2, n, p1, p2);
589 printk(" VGA1: clock is %d\n", CALC_VCLOCK(m1, m2, n, p1, p2)); 613 printk(" VGA1: clock is %d\n", calc_vclock(index, m1, m2, n, p1, p2));
590 614
591 printk(" DPLL_A: 0x%08x\n", hw->dpll_a); 615 printk(" DPLL_A: 0x%08x\n", hw->dpll_a);
592 printk(" DPLL_B: 0x%08x\n", hw->dpll_b); 616 printk(" DPLL_B: 0x%08x\n", hw->dpll_b);
593 printk(" FPA0: 0x%08x\n", hw->fpa0); 617 printk(" FPA0: 0x%08x\n", hw->fpa0);
594 printk(" FPA1: 0x%08x\n", hw->fpa1); 618 printk(" FPA1: 0x%08x\n", hw->fpa1);
595 printk(" FPB0: 0x%08x\n", hw->fpb0); 619 printk(" FPB0: 0x%08x\n", hw->fpb0);
596 printk(" FPB1: 0x%08x\n", hw->fpb1); 620 printk(" FPB1: 0x%08x\n", hw->fpb1);
597 621
598 n = (hw->fpa0 >> FP_N_DIVISOR_SHIFT) & FP_DIVISOR_MASK; 622 n = (hw->fpa0 >> FP_N_DIVISOR_SHIFT) & FP_DIVISOR_MASK;
599 m1 = (hw->fpa0 >> FP_M1_DIVISOR_SHIFT) & FP_DIVISOR_MASK; 623 m1 = (hw->fpa0 >> FP_M1_DIVISOR_SHIFT) & FP_DIVISOR_MASK;
600 m2 = (hw->fpa0 >> FP_M2_DIVISOR_SHIFT) & FP_DIVISOR_MASK; 624 m2 = (hw->fpa0 >> FP_M2_DIVISOR_SHIFT) & FP_DIVISOR_MASK;
@@ -604,9 +628,9 @@ intelfbhw_print_hw_state(struct intelfb_info *dinfo, struct intelfb_hwstate *hw)
604 p1 = (hw->dpll_a >> DPLL_P1_SHIFT) & DPLL_P1_MASK; 628 p1 = (hw->dpll_a >> DPLL_P1_SHIFT) & DPLL_P1_MASK;
605 p2 = (hw->dpll_a >> DPLL_P2_SHIFT) & DPLL_P2_MASK; 629 p2 = (hw->dpll_a >> DPLL_P2_SHIFT) & DPLL_P2_MASK;
606 printk(" PLLA0: (m1, m2, n, p1, p2) = (%d, %d, %d, %d, %d)\n", 630 printk(" PLLA0: (m1, m2, n, p1, p2) = (%d, %d, %d, %d, %d)\n",
607 m1, m2, n, p1, p2); 631 m1, m2, n, p1, p2);
608 printk(" PLLA0: clock is %d\n", CALC_VCLOCK(m1, m2, n, p1, p2)); 632 printk(" PLLA0: clock is %d\n", calc_vclock(index, m1, m2, n, p1, p2));
609 633
610 n = (hw->fpa1 >> FP_N_DIVISOR_SHIFT) & FP_DIVISOR_MASK; 634 n = (hw->fpa1 >> FP_N_DIVISOR_SHIFT) & FP_DIVISOR_MASK;
611 m1 = (hw->fpa1 >> FP_M1_DIVISOR_SHIFT) & FP_DIVISOR_MASK; 635 m1 = (hw->fpa1 >> FP_M1_DIVISOR_SHIFT) & FP_DIVISOR_MASK;
612 m2 = (hw->fpa1 >> FP_M2_DIVISOR_SHIFT) & FP_DIVISOR_MASK; 636 m2 = (hw->fpa1 >> FP_M2_DIVISOR_SHIFT) & FP_DIVISOR_MASK;
@@ -616,16 +640,16 @@ intelfbhw_print_hw_state(struct intelfb_info *dinfo, struct intelfb_hwstate *hw)
616 p1 = (hw->dpll_a >> DPLL_P1_SHIFT) & DPLL_P1_MASK; 640 p1 = (hw->dpll_a >> DPLL_P1_SHIFT) & DPLL_P1_MASK;
617 p2 = (hw->dpll_a >> DPLL_P2_SHIFT) & DPLL_P2_MASK; 641 p2 = (hw->dpll_a >> DPLL_P2_SHIFT) & DPLL_P2_MASK;
618 printk(" PLLA1: (m1, m2, n, p1, p2) = (%d, %d, %d, %d, %d)\n", 642 printk(" PLLA1: (m1, m2, n, p1, p2) = (%d, %d, %d, %d, %d)\n",
619 m1, m2, n, p1, p2); 643 m1, m2, n, p1, p2);
620 printk(" PLLA1: clock is %d\n", CALC_VCLOCK(m1, m2, n, p1, p2)); 644 printk(" PLLA1: clock is %d\n", calc_vclock(index, m1, m2, n, p1, p2));
621 645
622#if 0 646#if 0
623 printk(" PALETTE_A:\n"); 647 printk(" PALETTE_A:\n");
624 for (i = 0; i < PALETTE_8_ENTRIES) 648 for (i = 0; i < PALETTE_8_ENTRIES)
625 printk(" %3d: 0x%08x\n", i, hw->palette_a[i]; 649 printk(" %3d: 0x%08x\n", i, hw->palette_a[i]);
626 printk(" PALETTE_B:\n"); 650 printk(" PALETTE_B:\n");
627 for (i = 0; i < PALETTE_8_ENTRIES) 651 for (i = 0; i < PALETTE_8_ENTRIES)
628 printk(" %3d: 0x%08x\n", i, hw->palette_b[i]; 652 printk(" %3d: 0x%08x\n", i, hw->palette_b[i]);
629#endif 653#endif
630 654
631 printk(" HTOTAL_A: 0x%08x\n", hw->htotal_a); 655 printk(" HTOTAL_A: 0x%08x\n", hw->htotal_a);
@@ -700,12 +724,12 @@ intelfbhw_print_hw_state(struct intelfb_info *dinfo, struct intelfb_hwstate *hw)
700 } 724 }
701 for (i = 0; i < 3; i++) { 725 for (i = 0; i < 3; i++) {
702 printk(" SWF3%d 0x%08x\n", i, 726 printk(" SWF3%d 0x%08x\n", i,
703 hw->swf3x[i]); 727 hw->swf3x[i]);
704 } 728 }
705 for (i = 0; i < 8; i++) 729 for (i = 0; i < 8; i++)
706 printk(" FENCE%d 0x%08x\n", i, 730 printk(" FENCE%d 0x%08x\n", i,
707 hw->fence[i]); 731 hw->fence[i]);
708 732
709 printk(" INSTPM 0x%08x\n", hw->instpm); 733 printk(" INSTPM 0x%08x\n", hw->instpm);
710 printk(" MEM_MODE 0x%08x\n", hw->mem_mode); 734 printk(" MEM_MODE 0x%08x\n", hw->mem_mode);
711 printk(" FW_BLC_0 0x%08x\n", hw->fw_blc_0); 735 printk(" FW_BLC_0 0x%08x\n", hw->fw_blc_0);
@@ -715,6 +739,8 @@ intelfbhw_print_hw_state(struct intelfb_info *dinfo, struct intelfb_hwstate *hw)
715#endif 739#endif
716} 740}
717 741
742
743
718/* Split the M parameter into M1 and M2. */ 744/* Split the M parameter into M1 and M2. */
719static int 745static int
720splitm(int index, unsigned int m, unsigned int *retm1, unsigned int *retm2) 746splitm(int index, unsigned int m, unsigned int *retm1, unsigned int *retm2)
@@ -742,7 +768,17 @@ splitp(int index, unsigned int p, unsigned int *retp1, unsigned int *retp2)
742{ 768{
743 int p1, p2; 769 int p1, p2;
744 770
745 if (index==PLLS_I8xx) 771 if (index == PLLS_I9xx)
772 {
773 p1 = (p / 10) + 1;
774 p2 = 0;
775
776 *retp1 = (unsigned int)p1;
777 *retp2 = (unsigned int)p2;
778 return 0;
779 }
780
781 if (index == PLLS_I8xx)
746 { 782 {
747 if (p % 4 == 0) 783 if (p % 4 == 0)
748 p2 = 1; 784 p2 = 1;
@@ -812,7 +848,7 @@ calc_pll_params(int index, int clock, u32 *retm1, u32 *retm2, u32 *retn, u32 *re
812 m = plls[index].min_m; 848 m = plls[index].min_m;
813 if (m > plls[index].max_m) 849 if (m > plls[index].max_m)
814 m = plls[index].max_m; 850 m = plls[index].max_m;
815 f_out = CALC_VCLOCK3(m, n, p); 851 f_out = calc_vclock3(index, m, n, p);
816 if (splitm(index, m, &m1, &m2)) { 852 if (splitm(index, m, &m1, &m2)) {
817 WRN_MSG("cannot split m = %d\n", m); 853 WRN_MSG("cannot split m = %d\n", m);
818 n++; 854 n++;
@@ -849,14 +885,15 @@ calc_pll_params(int index, int clock, u32 *retm1, u32 *retm2, u32 *retn, u32 *re
849 DBG_MSG("m, n, p: %d (%d,%d), %d (%d), %d (%d,%d), " 885 DBG_MSG("m, n, p: %d (%d,%d), %d (%d), %d (%d,%d), "
850 "f: %d (%d), VCO: %d\n", 886 "f: %d (%d), VCO: %d\n",
851 m, m1, m2, n, n1, p, p1, p2, 887 m, m1, m2, n, n1, p, p1, p2,
852 CALC_VCLOCK3(m, n, p), CALC_VCLOCK(m1, m2, n1, p1, p2), 888 calc_vclock3(index, m, n, p),
853 CALC_VCLOCK3(m, n, p) * p); 889 calc_vclock(index, m1, m2, n1, p1, p2),
890 calc_vclock3(index, m, n, p) * p);
854 *retm1 = m1; 891 *retm1 = m1;
855 *retm2 = m2; 892 *retm2 = m2;
856 *retn = n1; 893 *retn = n1;
857 *retp1 = p1; 894 *retp1 = p1;
858 *retp2 = p2; 895 *retp2 = p2;
859 *retclock = CALC_VCLOCK(m1, m2, n1, p1, p2); 896 *retclock = calc_vclock(index, m1, m2, n1, p1, p2);
860 897
861 return 0; 898 return 0;
862} 899}
@@ -953,7 +990,7 @@ intelfbhw_mode_to_hw(struct intelfb_info *dinfo, struct intelfb_hwstate *hw,
953 /* Desired clock in kHz */ 990 /* Desired clock in kHz */
954 clock_target = 1000000000 / var->pixclock; 991 clock_target = 1000000000 / var->pixclock;
955 992
956 if (calc_pll_params(PLLS_I8xx, clock_target, &m1, &m2, &n, &p1, &p2, &clock)) { 993 if (calc_pll_params(dinfo->pll_index, clock_target, &m1, &m2, &n, &p1, &p2, &clock)) {
957 WRN_MSG("calc_pll_params failed\n"); 994 WRN_MSG("calc_pll_params failed\n");
958 return 1; 995 return 1;
959 } 996 }