nv04_timer.c 1.6 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283
  1. #include "drmP.h"
  2. #include "drm.h"
  3. #include "nouveau_drv.h"
  4. #include "nouveau_drm.h"
  5. int
  6. nv04_timer_init(struct drm_device *dev)
  7. {
  8. struct drm_nouveau_private *dev_priv = dev->dev_private;
  9. u32 m, n, d;
  10. nv_wr32(dev, NV04_PTIMER_INTR_EN_0, 0x00000000);
  11. nv_wr32(dev, NV04_PTIMER_INTR_0, 0xFFFFFFFF);
  12. /* aim for 31.25MHz, which gives us nanosecond timestamps */
  13. d = 1000000 / 32;
  14. /* determine base clock for timer source */
  15. if (dev_priv->chipset < 0x40) {
  16. n = dev_priv->engine.pm.clock_get(dev, PLL_CORE);
  17. } else
  18. if (dev_priv->chipset == 0x40) {
  19. /*XXX: figure this out */
  20. n = 0;
  21. } else {
  22. n = dev_priv->crystal;
  23. m = 1;
  24. while (n < (d * 2)) {
  25. n += (n / m);
  26. m++;
  27. }
  28. nv_wr32(dev, 0x009220, m - 1);
  29. }
  30. if (!n) {
  31. NV_WARN(dev, "PTIMER: unknown input clock freq\n");
  32. if (!nv_rd32(dev, NV04_PTIMER_NUMERATOR) ||
  33. !nv_rd32(dev, NV04_PTIMER_DENOMINATOR)) {
  34. nv_wr32(dev, NV04_PTIMER_NUMERATOR, 1);
  35. nv_wr32(dev, NV04_PTIMER_DENOMINATOR, 1);
  36. }
  37. return 0;
  38. }
  39. /* reduce ratio to acceptable values */
  40. while (((n % 5) == 0) && ((d % 5) == 0)) {
  41. n /= 5;
  42. d /= 5;
  43. }
  44. while (((n % 2) == 0) && ((d % 2) == 0)) {
  45. n /= 2;
  46. d /= 2;
  47. }
  48. while (n > 0xffff || d > 0xffff) {
  49. n >>= 1;
  50. d >>= 1;
  51. }
  52. nv_wr32(dev, NV04_PTIMER_NUMERATOR, n);
  53. nv_wr32(dev, NV04_PTIMER_DENOMINATOR, d);
  54. return 0;
  55. }
  56. u64
  57. nv04_timer_read(struct drm_device *dev)
  58. {
  59. u32 hi, lo;
  60. do {
  61. hi = nv_rd32(dev, NV04_PTIMER_TIME_1);
  62. lo = nv_rd32(dev, NV04_PTIMER_TIME_0);
  63. } while (hi != nv_rd32(dev, NV04_PTIMER_TIME_1));
  64. return ((u64)hi << 32 | lo);
  65. }
  66. void
  67. nv04_timer_takedown(struct drm_device *dev)
  68. {
  69. }