mpc86x_clk.c 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202
  1. /*
  2. * (C) Copyright 2003 Intracom S.A.
  3. * Pantelis Antoniou <panto@intracom.gr>
  4. *
  5. * This little program makes an exhaustive search for the
  6. * correct terms of pdf, mfi, mfn, mfd, s, dbrmo, in PLPRCR.
  7. * The goal is to produce a gclk2 from a xin input, while respecting
  8. * all the restrictions on their combination.
  9. *
  10. * Generaly you select the first row of the produced table.
  11. *
  12. * SPDX-License-Identifier: GPL-2.0+
  13. */
  14. #include <stdio.h>
  15. #include <stdlib.h>
  16. #define DPREF_MIN 10000000
  17. #define DPREF_MAX 32000000
  18. #define DPGDCK_MAX 320000000
  19. #define DPGDCK_MIN 160000000
  20. #define S_MIN 0
  21. #define S_MAX 2
  22. #define MFI_MIN 5
  23. #define MFI_MAX 15
  24. #define MFN_MIN 0
  25. #define MFN_MAX 15
  26. #define MFD_MIN 0
  27. #define MFD_MAX 31
  28. #define MF_MIN 5
  29. #define MF_MAX 15
  30. #define PDF_MIN 0
  31. #define PDF_MAX 15
  32. #define GCLK2_MAX 150000000
  33. static int calculate (int xin, int target_clock,
  34. int ppm, int pdf, int mfi, int mfn, int mfd, int s,
  35. int *dprefp, int *dpgdckp, int *jdbckp,
  36. int *gclk2p, int *dbrmop)
  37. {
  38. unsigned int dpref, dpgdck, jdbck, gclk2, t1, t2, dbrmo;
  39. /* valid MFI? */
  40. if (mfi < MFI_MIN)
  41. return -1;
  42. /* valid num, denum? */
  43. if (mfn > 0 && mfn >= mfd)
  44. return -1;
  45. dpref = xin / (pdf + 1);
  46. /* valid dpef? */
  47. if (dpref < DPREF_MIN || dpref > DPREF_MAX)
  48. return -1;
  49. if (mfn == 0) {
  50. dpgdck = (2 * mfi * xin) / (pdf + 1) ;
  51. dbrmo = 0;
  52. } else {
  53. /* 5 <= mfi + (mfn / mfd + 1) <= 15 */
  54. t1 = mfd + 1;
  55. t2 = mfi * t1 + mfn;
  56. if ( MF_MIN * t1 > t2 || MF_MAX * t1 < t2)
  57. return -1;
  58. dpgdck = (unsigned int)(2 * (mfi * mfd + mfi + mfn) *
  59. (unsigned int)xin) /
  60. ((mfd + 1) * (pdf + 1));
  61. dbrmo = 10 * mfn < (mfd + 1);
  62. }
  63. /* valid dpgclk? */
  64. if (dpgdck < DPGDCK_MIN || dpgdck > DPGDCK_MAX)
  65. return -1;
  66. jdbck = dpgdck >> s;
  67. gclk2 = jdbck / 2;
  68. /* valid gclk2 */
  69. if (gclk2 > GCLK2_MAX)
  70. return -1;
  71. t1 = abs(gclk2 - target_clock);
  72. /* XXX max 1MHz dev. in clock */
  73. if (t1 > 1000000)
  74. return -1;
  75. /* dev within range (XXX gclk2 scaled to avoid overflow) */
  76. if (t1 * 1000 > (unsigned int)ppm * (gclk2 / 1000))
  77. return -1;
  78. *dprefp = dpref;
  79. *dpgdckp = dpgdck;
  80. *jdbckp = jdbck;
  81. *gclk2p = gclk2;
  82. *dbrmop = dbrmo;
  83. return gclk2;
  84. }
  85. int conf_clock(int xin, int target_clock, int ppm)
  86. {
  87. int pdf, s, mfn, mfd, mfi;
  88. int dpref, dpgdck, jdbck, gclk2, xout, dbrmo;
  89. int found = 0;
  90. /* integer multipliers */
  91. for (pdf = PDF_MIN; pdf <= PDF_MAX; pdf++) {
  92. for (mfi = MFI_MIN; mfi <= MFI_MAX; mfi++) {
  93. for (s = 0; s <= S_MAX; s++) {
  94. xout = calculate(xin, target_clock,
  95. ppm, pdf, mfi, 0, 0, s,
  96. &dpref, &dpgdck, &jdbck,
  97. &gclk2, &dbrmo);
  98. if (xout < 0)
  99. continue;
  100. if (found == 0) {
  101. printf("pdf mfi mfn mfd s dbrmo dpref dpgdck jdbck gclk2 exact?\n");
  102. printf("--- --- --- --- - ----- ----- ------ ----- ----- ------\n");
  103. }
  104. printf("%3d %3d --- --- %1d %5d %9d %9d %9d %9d%s\n",
  105. pdf, mfi, s, dbrmo,
  106. dpref, dpgdck, jdbck, gclk2,
  107. gclk2 == target_clock ? " YES" : "");
  108. found++;
  109. }
  110. }
  111. }
  112. /* fractional multipliers */
  113. for (pdf = PDF_MIN; pdf <= PDF_MAX; pdf++) {
  114. for (mfi = MFI_MIN; mfi <= MFI_MAX; mfi++) {
  115. for (mfn = 1; mfn <= MFN_MAX; mfn++) {
  116. for (mfd = 1; mfd <= MFD_MAX; mfd++) {
  117. for (s = 0; s <= S_MAX; s++) {
  118. xout = calculate(xin, target_clock,
  119. ppm, pdf, mfi, mfn, mfd, s,
  120. &dpref, &dpgdck, &jdbck,
  121. &gclk2, &dbrmo);
  122. if (xout < 0)
  123. continue;
  124. if (found == 0) {
  125. printf("pdf mfi mfn mfd s dbrmo dpref dpgdck jdbck gclk2 exact?\n");
  126. printf("--- --- --- --- - ----- ----- ------ ----- ----- ------\n");
  127. }
  128. printf("%3d %3d %3d %3d %1d %5d %9d %9d %9d %9d%s\n",
  129. pdf, mfi, mfn, mfd, s,
  130. dbrmo, dpref, dpgdck, jdbck, gclk2,
  131. gclk2 == target_clock ? " YES" : "");
  132. found++;
  133. }
  134. }
  135. }
  136. }
  137. }
  138. return found;
  139. }
  140. int main(int argc, char *argv[])
  141. {
  142. int xin, want_gclk2, found, ppm = 100;
  143. if (argc < 3) {
  144. fprintf(stderr, "usage: mpc86x_clk <xin> <want_gclk2> [ppm]\n");
  145. fprintf(stderr, " default ppm is 100\n");
  146. return 10;
  147. }
  148. xin = atoi(argv[1]);
  149. want_gclk2 = atoi(argv[2]);
  150. if (argc >= 4)
  151. ppm = atoi(argv[3]);
  152. found = conf_clock(xin, want_gclk2, ppm);
  153. if (found <= 0) {
  154. fprintf(stderr, "cannot produce gclk2 %d from xin %d\n",
  155. want_gclk2, xin);
  156. return EXIT_FAILURE;
  157. }
  158. return EXIT_SUCCESS;
  159. }