2009-12-11 17:24:15 +08:00
|
|
|
#include "drmP.h"
|
|
|
|
#include "drm.h"
|
|
|
|
#include "nouveau_drv.h"
|
|
|
|
#include "nouveau_drm.h"
|
2011-10-27 08:28:17 +08:00
|
|
|
#include "nouveau_hw.h"
|
2009-12-11 17:24:15 +08:00
|
|
|
|
|
|
|
int
|
|
|
|
nv04_timer_init(struct drm_device *dev)
|
|
|
|
{
|
2011-07-03 19:16:12 +08:00
|
|
|
struct drm_nouveau_private *dev_priv = dev->dev_private;
|
|
|
|
u32 m, n, d;
|
|
|
|
|
2009-12-11 17:24:15 +08:00
|
|
|
nv_wr32(dev, NV04_PTIMER_INTR_EN_0, 0x00000000);
|
|
|
|
nv_wr32(dev, NV04_PTIMER_INTR_0, 0xFFFFFFFF);
|
|
|
|
|
2011-07-03 19:16:12 +08:00
|
|
|
/* aim for 31.25MHz, which gives us nanosecond timestamps */
|
2011-07-21 14:12:58 +08:00
|
|
|
d = 1000000 / 32;
|
2011-07-03 19:16:12 +08:00
|
|
|
|
|
|
|
/* determine base clock for timer source */
|
|
|
|
if (dev_priv->chipset < 0x40) {
|
2011-10-27 08:28:17 +08:00
|
|
|
n = nouveau_hw_get_clock(dev, PLL_CORE);
|
2011-07-03 19:16:12 +08:00
|
|
|
} else
|
|
|
|
if (dev_priv->chipset == 0x40) {
|
|
|
|
/*XXX: figure this out */
|
|
|
|
n = 0;
|
|
|
|
} else {
|
2011-07-21 14:12:58 +08:00
|
|
|
n = dev_priv->crystal;
|
2011-07-03 19:16:12 +08:00
|
|
|
m = 1;
|
|
|
|
while (n < (d * 2)) {
|
|
|
|
n += (n / m);
|
|
|
|
m++;
|
|
|
|
}
|
|
|
|
|
|
|
|
nv_wr32(dev, 0x009220, m - 1);
|
2009-12-11 17:24:15 +08:00
|
|
|
}
|
|
|
|
|
2011-07-03 19:16:12 +08:00
|
|
|
if (!n) {
|
|
|
|
NV_WARN(dev, "PTIMER: unknown input clock freq\n");
|
|
|
|
if (!nv_rd32(dev, NV04_PTIMER_NUMERATOR) ||
|
|
|
|
!nv_rd32(dev, NV04_PTIMER_DENOMINATOR)) {
|
|
|
|
nv_wr32(dev, NV04_PTIMER_NUMERATOR, 1);
|
|
|
|
nv_wr32(dev, NV04_PTIMER_DENOMINATOR, 1);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* reduce ratio to acceptable values */
|
|
|
|
while (((n % 5) == 0) && ((d % 5) == 0)) {
|
|
|
|
n /= 5;
|
|
|
|
d /= 5;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (((n % 2) == 0) && ((d % 2) == 0)) {
|
|
|
|
n /= 2;
|
|
|
|
d /= 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (n > 0xffff || d > 0xffff) {
|
|
|
|
n >>= 1;
|
|
|
|
d >>= 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
nv_wr32(dev, NV04_PTIMER_NUMERATOR, n);
|
|
|
|
nv_wr32(dev, NV04_PTIMER_DENOMINATOR, d);
|
2009-12-11 17:24:15 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2011-07-03 19:16:12 +08:00
|
|
|
u64
|
2009-12-11 17:24:15 +08:00
|
|
|
nv04_timer_read(struct drm_device *dev)
|
|
|
|
{
|
2011-07-03 19:16:12 +08:00
|
|
|
u32 hi, lo;
|
|
|
|
|
2009-12-11 17:24:15 +08:00
|
|
|
do {
|
2011-07-03 19:16:12 +08:00
|
|
|
hi = nv_rd32(dev, NV04_PTIMER_TIME_1);
|
|
|
|
lo = nv_rd32(dev, NV04_PTIMER_TIME_0);
|
|
|
|
} while (hi != nv_rd32(dev, NV04_PTIMER_TIME_1));
|
|
|
|
|
|
|
|
return ((u64)hi << 32 | lo);
|
2009-12-11 17:24:15 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
nv04_timer_takedown(struct drm_device *dev)
|
|
|
|
{
|
|
|
|
}
|