NetBSD/sys/dev/pci/radeonfb_bios.c
gdamore 0d7e99a27a Initial import of radeonfb. Works to some degree on at least some hardware.
Most of the testing was done on MIPS hardware -- it probably needs work before
it will be useful with x86 hardware, and it is probably incompatible with
the X11 server.

"ATI Technologies Inc. ("ATI") has not assisted in the creation of, and
does not endorse, this software.  ATI will not be responsible or liable
for any actual or alleged damage or loss caused by or in connection with
the use of or reliance on this software."

Enjoy!
2006-08-16 22:46:44 +00:00

620 lines
15 KiB
C

/* $NetBSD: radeonfb_bios.c,v 1.1 2006/08/16 22:46:45 gdamore Exp $ */
/*-
* Copyright (c) 2006 Itronix Inc.
* All rights reserved.
*
* Written by Garrett D'Amore for Itronix Inc.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* 3. The name of Itronix Inc. may not be used to endorse
* or promote products derived from this software without specific
* prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY ITRONIX INC. ``AS IS'' AND
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL ITRONIX INC. BE LIABLE FOR ANY
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
* ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGE.
*/
/*
* ATI Technologies Inc. ("ATI") has not assisted in the creation of, and
* does not endorse, this software. ATI will not be responsible or liable
* for any actual or alleged damage or loss caused by or in connection with
* the use of or reliance on this software.
*/
#include <sys/cdefs.h>
__KERNEL_RCSID(0, "$NetBSD: radeonfb_bios.c,v 1.1 2006/08/16 22:46:45 gdamore Exp $");
#include <sys/param.h>
#include <sys/systm.h>
#include <sys/device.h>
#include <sys/malloc.h>
#include <machine/bus.h>
#include <dev/pci/pcidevs.h>
#include <dev/pci/pcireg.h>
#include <dev/pci/pcivar.h>
#include <dev/pci/radeonfbreg.h>
#include <dev/pci/radeonfbvar.h>
#ifdef RADEON_BIOS_INIT
/*
* Globals for the entire BIOS.
*/
#define ROM_HEADER_OFFSET 0x48
#define MAX_REVISION 0x10
#define SINGLE_TABLE_REVISION 0x09
#define MIN_OFFSET 0x60
/*
* Offsets of specific tables.
*/
#define RAGE_REGS1_OFFSET 0x0c
#define RAGE_REGS2_OFFSET 0x4e
#define DYN_CLOCK_OFFSET 0x52
#define PLL_INIT_OFFSET 0x46
#define MEM_CONFIG_OFFSET 0x48
/*
* Values related to generic intialization tables.
*/
#define TABLE_ENTRY_FLAG_MASK 0xe000
#define TABLE_ENTRY_INDEX_MASK 0x1fff
#define TABLE_ENTRY_COMMAND_MASK 0x00ff
#define TABLE_FLAG_WRITE_INDEXED 0x0000
#define TABLE_FLAG_WRITE_DIRECT 0x2000
#define TABLE_FLAG_MASK_INDEXED 0x4000
#define TABLE_FLAG_MASK_DIRECT 0x6000
#define TABLE_FLAG_DELAY 0x8000
#define TABLE_FLAG_SCOMMAND 0xa000
#define TABLE_SCOMMAND_WAIT_MC_BUSY_MASK 0x03
#define TABLE_SCOMMAND_WAIT_MEM_PWRUP_COMPLETE 0x08
/*
* PLL initialization block values.
*/
#define PLL_FLAG_MASK 0xc0
#define PLL_INDEX_MASK 0x3f
#define PLL_FLAG_WRITE 0x00
#define PLL_FLAG_MASK_BYTE 0x40
#define PLL_FLAG_WAIT 0x80
#define PLL_WAIT_150MKS 1
#define PLL_WAIT_5MS 2
#define PLL_WAIT_MC_BUSY_MASK 3
#define PLL_WAIT_DLL_READY_MASK 4
#define PLL_WAIT_CHK_SET_CLK_PWRMGT_CNTL24 5
#ifdef RADEON_BIOS_DEBUG
#define DPRINTF(x) printf x
#else
#define DPRINTF(x)
#endif
struct rb_table;
static void rb_validate(struct radeonfb_softc *, struct rb_table *);
static uint16_t rb_find_asic_table(struct radeonfb_softc *, struct rb_table *);
static uint16_t rb_find_mem_reset_table(struct radeonfb_softc *,
struct rb_table *);
static uint16_t rb_find_short_mem_reset_table(struct radeonfb_softc *,
struct rb_table *);
static int rb_load_init_block(struct radeonfb_softc *, struct rb_table *);
static int rb_load_pll_block(struct radeonfb_softc *, struct rb_table *);
static int rb_reset_sdram(struct radeonfb_softc *, struct rb_table *);
static void rb_wait_mc_busy_mask(struct radeonfb_softc *, uint16_t);
static void rb_wait_mem_pwrup_complete(struct radeonfb_softc *, uint16_t);
static void rb_wait_dll_ready_mask(struct radeonfb_softc *, uint16_t);
static void rb_wait_chk_set_clk_pwrmgt_cntl24(struct radeonfb_softc *);
/*
* Generic structure describing the tables.
*/
struct rb_table {
const unsigned char *name;
uint16_t offset;
struct rb_table *parent;
/* validate that the table looks sane */
void (*validate)(struct radeonfb_softc *, struct rb_table *);
/* find looks for the table relative to its "parent" */
uint16_t (*find)(struct radeonfb_softc *, struct rb_table *);
};
/*
* Instances of specific tables.
*/
static struct rb_table rb_rage_regs1_table = {
"rage_regs_1", /* name */
RAGE_REGS1_OFFSET, /* offset */
NULL, /* parent */
rb_validate, /* validate */
NULL, /* find */
};
static struct rb_table rb_rage_regs2_table = {
"rage_regs_2", /* name */
RAGE_REGS2_OFFSET, /* offset */
NULL, /* parent */
rb_validate, /* validate */
NULL, /* find */
};
static struct rb_table rb_dyn_clock_table = {
"dyn_clock", /* name */
DYN_CLOCK_OFFSET, /* offset */
NULL, /* parent */
rb_validate, /* validate */
NULL, /* find */
};
static struct rb_table rb_pll_init_table = {
"pll_init", /* name */
PLL_INIT_OFFSET, /* offset */
NULL, /* parent */
rb_validate, /* validate */
NULL, /* find */
};
static struct rb_table rb_mem_config_table = {
"mem_config", /* name */
MEM_CONFIG_OFFSET, /* offset */
NULL, /* parent */
rb_validate, /* validate */
NULL, /* find */
};
static struct rb_table rb_mem_reset_table = {
"mem_reset", /* name */
0, /* offset */
&rb_mem_config_table, /* parent */
NULL, /* validate */
rb_find_mem_reset_table, /* find */
};
static struct rb_table rb_short_mem_reset_table = {
"short_mem_reset", /* name */
0, /* offset */
&rb_mem_config_table, /* parent */
NULL, /* validate */
rb_find_short_mem_reset_table, /* find */
};
static struct rb_table rb_rage_regs3_table = {
"rage_regs_3", /* name */
0, /* offset */
&rb_rage_regs2_table, /* parent */
NULL, /* validate */
rb_find_asic_table, /* find */
};
static struct rb_table rb_rage_regs4_table = {
"rage_regs_4", /* name */
0, /* offset */
&rb_rage_regs3_table, /* parent */
NULL, /* validate */
rb_find_asic_table, /* find */
};
static struct rb_table *rb_tables[] = {
&rb_rage_regs1_table,
&rb_rage_regs2_table,
&rb_dyn_clock_table,
&rb_pll_init_table,
&rb_mem_config_table,
&rb_mem_reset_table,
&rb_short_mem_reset_table,
&rb_rage_regs3_table,
&rb_rage_regs4_table,
NULL
};
void
rb_validate(struct radeonfb_softc *sc, struct rb_table *tp)
{
uint8_t rev;
rev = GETBIOS8(sc, tp->offset - 1);
if (rev > MAX_REVISION) {
DPRINTF(("%s: bad rev %x of %s\n", XNAME(sc), rev, tp->name));
tp->offset = 0;
return;
}
if (tp->offset < MIN_OFFSET) {
DPRINTF(("%s: wrong pointer to %s!\n", XNAME(sc), tp->name));
tp->offset = 0;
return;
}
}
uint16_t
rb_find_asic_table(struct radeonfb_softc *sc, struct rb_table *tp)
{
uint16_t offset;
uint8_t c;
if ((offset = tp->offset) != 0) {
while ((c = GETBIOS8(sc, offset + 1)) != 0) {
if (c & 0x40)
offset += 10;
else if (c & 0x80)
offset += 4;
else
offset += 6;
}
return offset + 2;
}
return 0;
}
uint16_t
rb_find_mem_reset_table(struct radeonfb_softc *sc, struct rb_table *tp)
{
uint16_t offset;
if ((offset = tp->offset) != 0) {
while (GETBIOS8(sc, offset))
offset++;
offset++;
return offset + 2; /* skip table revision and mask */
}
return 0;
}
uint16_t
rb_find_short_mem_reset_table(struct radeonfb_softc *sc, struct rb_table *tp)
{
if ((tp->offset != 0) && (GETBIOS8(sc, tp->offset - 2) <= 64))
return (tp->offset + GETBIOS8(sc, tp->offset - 3));
return 0;
}
/* helper commands */
void
rb_wait_mc_busy_mask(struct radeonfb_softc *sc, uint16_t count)
{
DPRINTF(("WAIT_MC_BUSY_MASK: %d ", count));
while (count--) {
if (!(radeonfb_getpll(sc, RADEON_CLK_PWRMGT_CNTL) &
RADEON_MC_BUSY_MASK))
break;
}
DPRINTF(("%d\n", count));
}
void
rb_wait_mem_pwrup_complete(struct radeonfb_softc *sc, uint16_t count)
{
DPRINTF(("WAIT_MEM_PWRUP_COMPLETE: %d ", count));
while (count--) {
if ((radeonfb_getindex(sc, RADEON_MEM_STR_CNTL) &
RADEON_MEM_PWRUP_COMPLETE) ==
RADEON_MEM_PWRUP_COMPLETE)
break;
}
DPRINTF(("%d\n", count));
}
void
rb_wait_dll_ready_mask(struct radeonfb_softc *sc, uint16_t count)
{
DPRINTF(("WAIT_DLL_READY_MASK: %d ", count));
while (count--) {
if (radeonfb_getpll(sc, RADEON_CLK_PWRMGT_CNTL) &
RADEON_DLL_READY_MASK)
break;
}
DPRINTF(("%d\n", count));
}
void
rb_wait_chk_set_clk_pwrmgt_cntl24(struct radeonfb_softc *sc)
{
uint32_t pmc;
DPRINTF(("WAIT CHK_SET_CLK_PWRMGT_CNTL24\n"));
pmc = radeonfb_getpll(sc, RADEON_CLK_PWRMGT_CNTL);
if (pmc & RADEON_CLK_PWRMGT_CNTL24) {
radeonfb_maskpll(sc, RADEON_MCLK_CNTL, 0xFFFF0000,
RADEON_SET_ALL_SRCS_TO_PCI);
delay(10000);
radeonfb_putpll(sc, RADEON_CLK_PWRMGT_CNTL,
pmc & ~RADEON_CLK_PWRMGT_CNTL24);
delay(10000);
}
}
/*
* Block initialization routines. These take action based on data in
* the tables.
*/
int
rb_load_init_block(struct radeonfb_softc *sc, struct rb_table *tp)
{
uint16_t offset;
uint16_t value;
if ((tp == NULL) || ((offset = tp->offset) == 0))
return 1;
DPRINTF(("%s: load_init_block processing %s\n", XNAME(sc), tp->name));
while ((value = GETBIOS16(sc, offset)) != 0) {
uint16_t flag = value & TABLE_ENTRY_FLAG_MASK;
uint16_t index = value & TABLE_ENTRY_INDEX_MASK;
uint8_t command = value & TABLE_ENTRY_COMMAND_MASK;
uint32_t ormask;
uint32_t andmask;
uint16_t count;
offset += 2;
switch (flag) {
case TABLE_FLAG_WRITE_INDEXED:
DPRINTF(("WRITE INDEXED: %x %x\n",
index, (uint32_t)GETBIOS32(sc, offset)));
radeonfb_putindex(sc, index, GETBIOS32(sc, offset));
offset += 4;
break;
case TABLE_FLAG_WRITE_DIRECT:
DPRINTF(("WRITE DIRECT: %x %x\n",
index, (uint32_t)GETBIOS32(sc, offset)));
radeonfb_put32(sc, index, GETBIOS32(sc, offset));
offset += 4;
break;
case TABLE_FLAG_MASK_INDEXED:
andmask = GETBIOS32(sc, offset);
offset += 4;
ormask = GETBIOS32(sc, offset);
offset += 4;
DPRINTF(("MASK INDEXED: %x %x %x\n",
index, andmask, ormask));
radeonfb_maskindex(sc, index, andmask, ormask);
break;
case TABLE_FLAG_MASK_DIRECT:
andmask = GETBIOS32(sc, offset);
offset += 4;
ormask = GETBIOS32(sc, offset);
offset += 4;
DPRINTF(("MASK DIRECT: %x %x %x\n",
index, andmask, ormask));
radeonfb_mask32(sc, index, andmask, ormask);
break;
case TABLE_FLAG_DELAY:
/* in the worst case, this would be 16msec */
count = GETBIOS16(sc, offset);
DPRINTF(("DELAY: %d\n", count));
delay(count);
offset += 2;
break;
case TABLE_FLAG_SCOMMAND:
DPRINTF(("SCOMMAND %x\n", command));
switch (command) {
case TABLE_SCOMMAND_WAIT_MC_BUSY_MASK:
count = GETBIOS16(sc, offset);
rb_wait_mc_busy_mask(sc, count);
break;
case TABLE_SCOMMAND_WAIT_MEM_PWRUP_COMPLETE:
count = GETBIOS16(sc, offset);
rb_wait_mem_pwrup_complete(sc, count);
break;
}
offset += 2;
break;
}
}
return 0;
}
int
rb_load_pll_block(struct radeonfb_softc *sc, struct rb_table *tp)
{
uint16_t offset;
uint8_t index;
uint8_t shift;
uint32_t andmask;
uint32_t ormask;
if ((tp == NULL) || ((offset = tp->offset) == 0))
return 1;
DPRINTF(("%s: load_pll_block processing %s\n", XNAME(sc), tp->name));
while ((index = GETBIOS8(sc, offset)) != 0) {
offset++;
switch (index & PLL_FLAG_MASK) {
case PLL_FLAG_WAIT:
switch (index & PLL_INDEX_MASK) {
case PLL_WAIT_150MKS:
delay(150);
break;
case PLL_WAIT_5MS:
/* perhaps this should be tsleep? */
delay(5000);
break;
case PLL_WAIT_MC_BUSY_MASK:
rb_wait_mc_busy_mask(sc, 1000);
break;
case PLL_WAIT_DLL_READY_MASK:
rb_wait_dll_ready_mask(sc, 1000);
break;
case PLL_WAIT_CHK_SET_CLK_PWRMGT_CNTL24:
rb_wait_chk_set_clk_pwrmgt_cntl24(sc);
break;
}
break;
case PLL_FLAG_MASK_BYTE:
shift = GETBIOS8(sc, offset) * 8;
offset++;
andmask =
(((uint32_t)GETBIOS8(sc, offset)) << shift) |
~((uint32_t)0xff << shift);
offset++;
ormask = ((uint32_t)GETBIOS8(sc, offset)) << shift;
offset++;
DPRINTF(("PLL_MASK_BYTE %u %u %x %x\n", index,
shift, andmask, ormask));
radeonfb_maskpll(sc, index, andmask, ormask);
break;
case PLL_FLAG_WRITE:
DPRINTF(("PLL_WRITE %u %x\n", index,
GETBIOS32(sc, offset)));
radeonfb_putpll(sc, index, GETBIOS32(sc, offset));
offset += 4;
break;
}
}
return 0;
}
int
rb_reset_sdram(struct radeonfb_softc *sc, struct rb_table *tp)
{
uint16_t offset;
uint8_t index;
if ((tp == NULL) || ((offset = tp->offset) == 0))
return 1;
DPRINTF(("%s: reset_sdram processing %s\n", XNAME(sc), tp->name));
while ((index = GETBIOS8(sc, offset)) != 0xff) {
offset++;
if (index == 0x0f) {
rb_wait_mem_pwrup_complete(sc, 20000);
} else {
uint32_t ormask;
ormask = GETBIOS16(sc, offset);
offset += 2;
DPRINTF(("INDEX reg RADEON_MEM_SDRAM_MODE_REG %x %x\n",
RADEON_SDRAM_MODE_MASK, ormask));
radeonfb_maskindex(sc, RADEON_MEM_SDRAM_MODE_REG,
RADEON_SDRAM_MODE_MASK, ormask);
ormask = (uint32_t)index << 24;
DPRINTF(("INDEX reg RADEON_MEM_SDRAM_MODE_REG %x %x\n",
RADEON_B3MEM_RESET_MASK, ormask));
radeonfb_maskindex(sc, RADEON_MEM_SDRAM_MODE_REG,
RADEON_B3MEM_RESET_MASK, ormask);
}
}
return 0;
}
/*
* Master entry point to parse and act on table data.
*/
int
radeonfb_bios_init(struct radeonfb_softc *sc)
{
uint16_t revision;
uint16_t scratch;
int i;
struct rb_table *tp;
if (!sc->sc_biossz)
return 1;
scratch = GETBIOS16(sc, ROM_HEADER_OFFSET);
revision = GETBIOS8(sc, scratch);
DPRINTF(("%s: Bios Rev: %d\n", XNAME(sc), revision));
/* First parse pass -- locate tables */
for (i = 0; (tp = rb_tables[i]) != NULL; i++) {
DPRINTF(("%s: parsing table %s\n", XNAME(sc), tp->name));
if (tp->offset != 0) {
uint16_t temp, offset;;
temp = GETBIOS16(sc, ROM_HEADER_OFFSET);
offset = GETBIOS16(sc, temp + tp->offset);
if (offset)
tp->offset = offset;
} else {
tp->offset = tp->find(sc, tp->parent);
}
if (tp->validate)
tp->validate(sc, tp);
if (revision > SINGLE_TABLE_REVISION)
break;
}
if (rb_rage_regs3_table.offset + 1 == rb_pll_init_table.offset) {
rb_rage_regs3_table.offset = 0;
rb_rage_regs4_table.offset = 0;
}
if (rb_rage_regs1_table.offset)
rb_load_init_block(sc, &rb_rage_regs1_table);
if (revision < SINGLE_TABLE_REVISION) {
if (rb_pll_init_table.offset)
rb_load_pll_block(sc, &rb_pll_init_table);
if (rb_rage_regs2_table.offset)
rb_load_init_block(sc, &rb_rage_regs2_table);
if (rb_rage_regs4_table.offset)
rb_load_init_block(sc, &rb_rage_regs4_table);
if (rb_mem_reset_table.offset)
rb_reset_sdram(sc, &rb_mem_reset_table);
if (rb_rage_regs3_table.offset)
rb_load_init_block(sc, &rb_rage_regs3_table);
if (rb_dyn_clock_table.offset)
rb_load_pll_block(sc, &rb_dyn_clock_table);
}
DPRINTF(("%s: BIOS parse done\n", XNAME(sc)));
return 0;
}
#endif