#define MODULE_BIT 0x00010000
#include "nm_std.h"
status_t nm_dac_mode(int mode,float brightness)
{
uint8 *r, *g, *b, t[256];
uint16 i;
r = si->color_data;
g = r + 256;
b = g + 256;
LOG(4,("DAC: Setting screen mode %d brightness %f\n", mode, brightness));
for (i = 0; i < 256; i++)
{
int ri = i * brightness;
if (ri > 255) ri = 255;
t[i] = ri;
}
switch(mode)
{
case BPP8:
LOG(8,("DAC: 8bit mode is indexed by OS, aborting brightness setup.\n"));
return B_OK;
break;
case BPP24:
LOG(8,("DAC: 24bit mode is a direct mode, aborting brightness setup.\n"));
return B_OK;
break;
case BPP16:
for (i = 0; i < 32; i++)
{
b[i] = r[i] = t[i << 3];
}
for (i = 0; i < 64; i++)
{
g[i] = t[i << 2];
}
break;
case BPP15:
for (i = 0; i < 32; i++)
{
g[i] = r[i] = b[i] = t[i << 3];
}
break;
default:
LOG(8,("DAC: Invalid colordepth requested, aborting!\n"));
return B_ERROR;
break;
}
if (nm_dac_palette(r, g, b, i) != B_OK) return B_ERROR;
ISAWB(PALMASK, 0xff);
LOG(2,("DAC: PAL pixrdmsk readback $%02x\n", ISARB(PALMASK)));
return B_OK;
}
status_t nm_dac_palette(uint8 r[256],uint8 g[256],uint8 b[256], uint16 cnt)
{
int i;
LOG(4,("DAC: setting palette\n"));
ISAWB(PALINDW, 0x00);
for (i = 0; i < cnt; i++)
{
ISAWB(PALDATA, (r[i] >> 2));
ISAWB(PALDATA, (g[i] >> 2));
ISAWB(PALDATA, (b[i] >> 2));
}
if (ISARB(PALINDW) != (cnt & 0x00ff))
{
LOG(8,("DAC: PAL write index incorrect after programming\n"));
return B_ERROR;
}
if (1)
{
uint8 R, G, B;
ISAWB(PALINDR, 0x00);
for (i = 0; i < cnt; i++)
{
R = (ISARB(PALDATA) << 2);
G = (ISARB(PALDATA) << 2);
B = (ISARB(PALDATA) << 2);
if (((r[i] & 0xfc) != R) || ((g[i] & 0xfc) != G) || ((b[i] & 0xfc) != B))
LOG(1,("DAC palette %d: w %x %x %x, r %x %x %x\n", i, r[i], g[i], b[i], R, G, B));
}
}
return B_OK;
}
status_t nm_dac_set_pix_pll(display_mode target)
{
uint8 m=0,n=0,p=0;
uint8 temp;
float pix_setting, req_pclk;
status_t result;
req_pclk = (target.timing.pixel_clock)/1000.0;
LOG(4,("DAC: Setting PIX PLL for pixelclock %f\n", req_pclk));
result = nm_dac_pix_pll_find(target,&pix_setting,&m,&n,&p);
if (result != B_OK)
{
return result;
}
temp = (ISARB(MISCR) | 0x0c);
snooze(10);
ISAWB(MISCW, temp);
ISAGRPHW(PLLC_NL, n);
if (si->ps.card_type >= NM2200)
{
temp = (ISAGRPHR(PLLC_NH) & 0x0f);
snooze(10);
ISAGRPHW(PLLC_NH, (temp | (p & 0xf0)));
}
ISAGRPHW(PLLC_M, m);
snooze(1000);
LOG(4,("DAC: PLLSEL $%02x\n", ISARB(MISCR)));
LOG(4,("DAC: PLLN $%02x\n", ISAGRPHR(PLLC_NL)));
LOG(4,("DAC: PLLM $%02x\n", ISAGRPHR(PLLC_M)));
LOG(2,("DAC: PIX PLL frequency should be locked now...\n"));
return B_OK;
}
status_t nm_dac_pix_pll_find
(display_mode target,float * calc_pclk,uint8 * m_result,uint8 * n_result,uint8 * p_result)
{
int m = 0, n = 0, p = 0, n_max, m_max;
float error, error_best = 999999999;
int best[2] = { 0, 0 };
float f_vco, max_pclk;
float req_pclk = target.timing.pixel_clock/1000.0;
switch (si->ps.card_type)
{
case NM2070:
LOG(4,("DAC: NM2070 restrictions apply\n"));
m_max = 32;
n_max = 128;
break;
case NM2090:
case NM2093:
case NM2097:
case NM2160:
LOG(4,("DAC: NM2090/93/97/NM2160 restrictions apply\n"));
m_max = 64;
n_max = 128;
break;
default:
LOG(4,("DAC: NM22xx/NM23xx restrictions apply\n"));
m_max = 64;
n_max = 2048;
break;
}
switch (target.space)
{
case B_CMAP8:
max_pclk = si->ps.max_dac1_clock_8;
break;
case B_RGB15_LITTLE:
case B_RGB16_LITTLE:
max_pclk = si->ps.max_dac1_clock_16;
break;
case B_RGB24_LITTLE:
max_pclk = si->ps.max_dac1_clock_24;
break;
default:
max_pclk = si->ps.max_dac1_clock_24;
break;
}
if (req_pclk < si->ps.min_pixel_vco)
{
LOG(4,("DAC: clamping pixclock: requested %fMHz, set to %fMHz\n",
req_pclk, (float)si->ps.min_pixel_vco));
req_pclk = si->ps.min_pixel_vco;
}
if (req_pclk > max_pclk)
{
LOG(4,("DAC: clamping pixclock: requested %fMHz, set to %fMHz\n",
req_pclk, (float)max_pclk));
req_pclk = max_pclk;
}
f_vco = req_pclk;
for (m = 1; m <= m_max; m++)
{
if ((m > (m_max / 2)) && ((m / 2.0) != 0.0)) continue;
n = (int)(((f_vco * m) / si->ps.f_ref) + 0.5);
if ((n < 1) || (n > n_max)) continue;
error = fabs(req_pclk - ((si->ps.f_ref / m) * n));
if (error < error_best)
{
error_best = error;
best[0] = m;
best[1] = n;
}
}
n = best[1] - 1;
if (best[0] <= (m_max / 2))
{
m = best[0] - 1;
p = (0 << 7);
}
else
{
m = (best[0] / 2) - 1;
p = (1 << 7);
}
*calc_pclk = (si->ps.f_ref / best[0]) * best[1];
*m_result = m;
if (si->ps.card_type < NM2200)
{
*n_result = ((n & 0x07f) | p);
*p_result = 0;
LOG(2,("DAC: pix PLL check: requested %fMHz got %fMHz, nm $%02x $%02x\n",
req_pclk, *calc_pclk, *n_result, *m_result));
}
else
{
*n_result = (n & 0x0ff);
*p_result = (((n & 0x700) >> 4) | p);
LOG(2,("DAC: pix PLL check: requested %fMHz got %fMHz, pnm $%02x $%02x $%02x\n",
req_pclk, *calc_pclk, *p_result, *n_result, *m_result));
}
return B_OK;
}