blob: afb9d4b6a02918fab8364e89c51abaf32e96ccea [file] [log] [blame]
Ben Skeggs6ee73862009-12-11 19:24:15 +10001#include "drmP.h"
2#include "drm.h"
3#include "nouveau_drv.h"
4#include "nouveau_drm.h"
5
Ben Skeggs591b06d2011-07-03 21:16:12 +10006static u32
7nv04_crystal_freq(struct drm_device *dev)
Ben Skeggs6ee73862009-12-11 19:24:15 +10008{
Ben Skeggs591b06d2011-07-03 21:16:12 +10009 struct drm_nouveau_private *dev_priv = dev->dev_private;
10 u32 extdev_boot0 = nv_rd32(dev, 0x101000);
11 int type;
Ben Skeggs6ee73862009-12-11 19:24:15 +100012
Ben Skeggs591b06d2011-07-03 21:16:12 +100013 type = !!(extdev_boot0 & 0x00000040);
14 if ((dev_priv->chipset >= 0x17 && dev_priv->chipset < 0x20) ||
15 dev_priv->chipset >= 0x25)
16 type |= (extdev_boot0 & 0x00400000) ? 2 : 0;
17
18 switch (type) {
19 case 0: return 13500000;
20 case 1: return 14318180;
21 case 2: return 27000000;
22 case 3: return 25000000;
23 default:
24 break;
Ben Skeggs6ee73862009-12-11 19:24:15 +100025 }
26
27 return 0;
28}
29
Ben Skeggs591b06d2011-07-03 21:16:12 +100030int
31nv04_timer_init(struct drm_device *dev)
32{
33 struct drm_nouveau_private *dev_priv = dev->dev_private;
34 u32 m, n, d;
35
36 nv_wr32(dev, NV04_PTIMER_INTR_EN_0, 0x00000000);
37 nv_wr32(dev, NV04_PTIMER_INTR_0, 0xFFFFFFFF);
38
39 /* aim for 31.25MHz, which gives us nanosecond timestamps */
40 d = 1000000000 / 32;
41
42 /* determine base clock for timer source */
43 if (dev_priv->chipset < 0x40) {
44 n = dev_priv->engine.pm.clock_get(dev, PLL_CORE);
45 } else
46 if (dev_priv->chipset == 0x40) {
47 /*XXX: figure this out */
48 n = 0;
49 } else {
50 n = nv04_crystal_freq(dev);
51 m = 1;
52 while (n < (d * 2)) {
53 n += (n / m);
54 m++;
55 }
56
57 nv_wr32(dev, 0x009220, m - 1);
58 }
59
60 if (!n) {
61 NV_WARN(dev, "PTIMER: unknown input clock freq\n");
62 if (!nv_rd32(dev, NV04_PTIMER_NUMERATOR) ||
63 !nv_rd32(dev, NV04_PTIMER_DENOMINATOR)) {
64 nv_wr32(dev, NV04_PTIMER_NUMERATOR, 1);
65 nv_wr32(dev, NV04_PTIMER_DENOMINATOR, 1);
66 }
67 return 0;
68 }
69
70 /* reduce ratio to acceptable values */
71 while (((n % 5) == 0) && ((d % 5) == 0)) {
72 n /= 5;
73 d /= 5;
74 }
75
76 while (((n % 2) == 0) && ((d % 2) == 0)) {
77 n /= 2;
78 d /= 2;
79 }
80
81 while (n > 0xffff || d > 0xffff) {
82 n >>= 1;
83 d >>= 1;
84 }
85
86 nv_wr32(dev, NV04_PTIMER_NUMERATOR, n);
87 nv_wr32(dev, NV04_PTIMER_DENOMINATOR, d);
88 return 0;
89}
90
91u64
Ben Skeggs6ee73862009-12-11 19:24:15 +100092nv04_timer_read(struct drm_device *dev)
93{
Ben Skeggs591b06d2011-07-03 21:16:12 +100094 u32 hi, lo;
95
Ben Skeggs6ee73862009-12-11 19:24:15 +100096 do {
Ben Skeggs591b06d2011-07-03 21:16:12 +100097 hi = nv_rd32(dev, NV04_PTIMER_TIME_1);
98 lo = nv_rd32(dev, NV04_PTIMER_TIME_0);
99 } while (hi != nv_rd32(dev, NV04_PTIMER_TIME_1));
100
101 return ((u64)hi << 32 | lo);
Ben Skeggs6ee73862009-12-11 19:24:15 +1000102}
103
104void
105nv04_timer_takedown(struct drm_device *dev)
106{
107}