mpc8536_serdes.c 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180
  1. /*
  2. * Copyright (C) 2008 Freescale Semicondutor, Inc. All rights reserved.
  3. * Dave Liu <daveliu@freescale.com>
  4. *
  5. * This program is free software; you can redistribute it and/or modify it
  6. * under the terms of the GNU General Public License as published by the
  7. * Free Software Foundation; either version 2 of the License, or (at your
  8. * option) any later version.
  9. */
  10. #include <config.h>
  11. #include <common.h>
  12. #include <asm/io.h>
  13. #include <asm/immap_85xx.h>
  14. /* PORDEVSR register */
  15. #define GUTS_PORDEVSR_OFFS 0xc
  16. #define GUTS_PORDEVSR_SERDES2_IO_SEL 0x38000000
  17. #define GUTS_PORDEVSR_SERDES2_IO_SEL_SHIFT 27
  18. /* SerDes CR0 register */
  19. #define FSL_SRDSCR0_OFFS 0x0
  20. #define FSL_SRDSCR0_TXEQA_MASK 0x00007000
  21. #define FSL_SRDSCR0_TXEQA_SGMII 0x00004000
  22. #define FSL_SRDSCR0_TXEQA_SATA 0x00001000
  23. #define FSL_SRDSCR0_TXEQE_MASK 0x00000700
  24. #define FSL_SRDSCR0_TXEQE_SGMII 0x00000400
  25. #define FSL_SRDSCR0_TXEQE_SATA 0x00000100
  26. /* SerDes CR1 register */
  27. #define FSL_SRDSCR1_OFFS 0x4
  28. #define FSL_SRDSCR1_LANEA_MASK 0x80200000
  29. #define FSL_SRDSCR1_LANEA_OFF 0x80200000
  30. #define FSL_SRDSCR1_LANEE_MASK 0x08020000
  31. #define FSL_SRDSCR1_LANEE_OFF 0x08020000
  32. /* SerDes CR2 register */
  33. #define FSL_SRDSCR2_OFFS 0x8
  34. #define FSL_SRDSCR2_EICA_MASK 0x00001f00
  35. #define FSL_SRDSCR2_EICA_SGMII 0x00000400
  36. #define FSL_SRDSCR2_EICA_SATA 0x00001400
  37. #define FSL_SRDSCR2_EICE_MASK 0x0000001f
  38. #define FSL_SRDSCR2_EICE_SGMII 0x00000004
  39. #define FSL_SRDSCR2_EICE_SATA 0x00000014
  40. /* SerDes CR3 register */
  41. #define FSL_SRDSCR3_OFFS 0xc
  42. #define FSL_SRDSCR3_LANEA_MASK 0x3f000700
  43. #define FSL_SRDSCR3_LANEA_SGMII 0x00000000
  44. #define FSL_SRDSCR3_LANEA_SATA 0x15000500
  45. #define FSL_SRDSCR3_LANEE_MASK 0x003f0007
  46. #define FSL_SRDSCR3_LANEE_SGMII 0x00000000
  47. #define FSL_SRDSCR3_LANEE_SATA 0x00150005
  48. void fsl_serdes_init(void)
  49. {
  50. void *guts = (void *)(CONFIG_SYS_MPC85xx_GUTS_ADDR);
  51. void *sd = (void *)CONFIG_SYS_MPC85xx_SERDES2_ADDR;
  52. u32 pordevsr = in_be32(guts + GUTS_PORDEVSR_OFFS);
  53. u32 srds2_io_sel;
  54. u32 tmp;
  55. /* parse the SRDS2_IO_SEL of PORDEVSR */
  56. srds2_io_sel = (pordevsr & GUTS_PORDEVSR_SERDES2_IO_SEL)
  57. >> GUTS_PORDEVSR_SERDES2_IO_SEL_SHIFT;
  58. switch (srds2_io_sel) {
  59. case 1: /* Lane A - SATA1, Lane E - SATA2 */
  60. /* CR 0 */
  61. tmp = in_be32(sd + FSL_SRDSCR0_OFFS);
  62. tmp &= ~FSL_SRDSCR0_TXEQA_MASK;
  63. tmp |= FSL_SRDSCR0_TXEQA_SATA;
  64. tmp &= ~FSL_SRDSCR0_TXEQE_MASK;
  65. tmp |= FSL_SRDSCR0_TXEQE_SATA;
  66. out_be32(sd + FSL_SRDSCR0_OFFS, tmp);
  67. /* CR 1 */
  68. tmp = in_be32(sd + FSL_SRDSCR1_OFFS);
  69. tmp &= ~FSL_SRDSCR1_LANEA_MASK;
  70. tmp &= ~FSL_SRDSCR1_LANEE_MASK;
  71. out_be32(sd + FSL_SRDSCR1_OFFS, tmp);
  72. /* CR 2 */
  73. tmp = in_be32(sd + FSL_SRDSCR2_OFFS);
  74. tmp &= ~FSL_SRDSCR2_EICA_MASK;
  75. tmp |= FSL_SRDSCR2_EICA_SATA;
  76. tmp &= ~FSL_SRDSCR2_EICE_MASK;
  77. tmp |= FSL_SRDSCR2_EICE_SATA;
  78. out_be32(sd + FSL_SRDSCR2_OFFS, tmp);
  79. /* CR 3 */
  80. tmp = in_be32(sd + FSL_SRDSCR3_OFFS);
  81. tmp &= ~FSL_SRDSCR3_LANEA_MASK;
  82. tmp |= FSL_SRDSCR3_LANEA_SATA;
  83. tmp &= ~FSL_SRDSCR3_LANEE_MASK;
  84. tmp |= FSL_SRDSCR3_LANEE_SATA;
  85. out_be32(sd + FSL_SRDSCR3_OFFS, tmp);
  86. break;
  87. case 3: /* Lane A - SATA1, Lane E - disabled */
  88. /* CR 0 */
  89. tmp = in_be32(sd + FSL_SRDSCR0_OFFS);
  90. tmp &= ~FSL_SRDSCR0_TXEQA_MASK;
  91. tmp |= FSL_SRDSCR0_TXEQA_SATA;
  92. out_be32(sd + FSL_SRDSCR0_OFFS, tmp);
  93. /* CR 1 */
  94. tmp = in_be32(sd + FSL_SRDSCR1_OFFS);
  95. tmp &= ~FSL_SRDSCR1_LANEE_MASK;
  96. tmp |= FSL_SRDSCR1_LANEE_OFF;
  97. out_be32(sd + FSL_SRDSCR1_OFFS, tmp);
  98. /* CR 2 */
  99. tmp = in_be32(sd + FSL_SRDSCR2_OFFS);
  100. tmp &= ~FSL_SRDSCR2_EICA_MASK;
  101. tmp |= FSL_SRDSCR2_EICA_SATA;
  102. out_be32(sd + FSL_SRDSCR2_OFFS, tmp);
  103. /* CR 3 */
  104. tmp = in_be32(sd + FSL_SRDSCR3_OFFS);
  105. tmp &= ~FSL_SRDSCR3_LANEA_MASK;
  106. tmp |= FSL_SRDSCR3_LANEA_SATA;
  107. out_be32(sd + FSL_SRDSCR3_OFFS, tmp);
  108. break;
  109. case 4: /* Lane A - eTSEC1 SGMII, Lane E - eTSEC3 SGMII */
  110. /* CR 0 */
  111. tmp = in_be32(sd + FSL_SRDSCR0_OFFS);
  112. tmp &= ~FSL_SRDSCR0_TXEQA_MASK;
  113. tmp |= FSL_SRDSCR0_TXEQA_SGMII;
  114. tmp &= ~FSL_SRDSCR0_TXEQE_MASK;
  115. tmp |= FSL_SRDSCR0_TXEQE_SGMII;
  116. out_be32(sd + FSL_SRDSCR0_OFFS, tmp);
  117. /* CR 1 */
  118. tmp = in_be32(sd + FSL_SRDSCR1_OFFS);
  119. tmp &= ~FSL_SRDSCR1_LANEA_MASK;
  120. tmp &= ~FSL_SRDSCR1_LANEE_MASK;
  121. out_be32(sd + FSL_SRDSCR1_OFFS, tmp);
  122. /* CR 2 */
  123. tmp = in_be32(sd + FSL_SRDSCR2_OFFS);
  124. tmp &= ~FSL_SRDSCR2_EICA_MASK;
  125. tmp |= FSL_SRDSCR2_EICA_SGMII;
  126. tmp &= ~FSL_SRDSCR2_EICE_MASK;
  127. tmp |= FSL_SRDSCR2_EICE_SGMII;
  128. out_be32(sd + FSL_SRDSCR2_OFFS, tmp);
  129. /* CR 3 */
  130. tmp = in_be32(sd + FSL_SRDSCR3_OFFS);
  131. tmp &= ~FSL_SRDSCR3_LANEA_MASK;
  132. tmp |= FSL_SRDSCR3_LANEA_SGMII;
  133. tmp &= ~FSL_SRDSCR3_LANEE_MASK;
  134. tmp |= FSL_SRDSCR3_LANEE_SGMII;
  135. out_be32(sd + FSL_SRDSCR3_OFFS, tmp);
  136. break;
  137. case 6: /* Lane A - eTSEC1 SGMII, Lane E - disabled */
  138. /* CR 0 */
  139. tmp = in_be32(sd + FSL_SRDSCR0_OFFS);
  140. tmp &= ~FSL_SRDSCR0_TXEQA_MASK;
  141. tmp |= FSL_SRDSCR0_TXEQA_SGMII;
  142. out_be32(sd + FSL_SRDSCR0_OFFS, tmp);
  143. /* CR 1 */
  144. tmp = in_be32(sd + FSL_SRDSCR1_OFFS);
  145. tmp &= ~FSL_SRDSCR1_LANEE_MASK;
  146. tmp |= FSL_SRDSCR1_LANEE_OFF;
  147. out_be32(sd + FSL_SRDSCR1_OFFS, tmp);
  148. /* CR 2 */
  149. tmp = in_be32(sd + FSL_SRDSCR2_OFFS);
  150. tmp &= ~FSL_SRDSCR2_EICA_MASK;
  151. tmp |= FSL_SRDSCR2_EICA_SGMII;
  152. out_be32(sd + FSL_SRDSCR2_OFFS, tmp);
  153. /* CR 3 */
  154. tmp = in_be32(sd + FSL_SRDSCR3_OFFS);
  155. tmp &= ~FSL_SRDSCR3_LANEA_MASK;
  156. tmp |= FSL_SRDSCR3_LANEA_SGMII;
  157. out_be32(sd + FSL_SRDSCR3_OFFS, tmp);
  158. break;
  159. case 7: /* Lane A - disabled, Lane E - disabled */
  160. /* CR 1 */
  161. tmp = in_be32(sd + FSL_SRDSCR1_OFFS);
  162. tmp &= ~FSL_SRDSCR1_LANEA_MASK;
  163. tmp |= FSL_SRDSCR1_LANEA_OFF;
  164. tmp &= ~FSL_SRDSCR1_LANEE_MASK;
  165. tmp |= FSL_SRDSCR1_LANEE_OFF;
  166. out_be32(sd + FSL_SRDSCR1_OFFS, tmp);
  167. break;
  168. default:
  169. break;
  170. }
  171. }