blob: 2016a87e2fe1c4b15b1a172a4691ac165f1597e8 [file] [log] [blame]
/*
* This file is part of the coreboot project.
*
* Copyright (C) 2007-2009 coresystems GmbH
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; version 2 of the License.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*/
#include <types.h>
#include <spd.h>
#include <spd_ddr2.h>
#include <delay.h>
#include "registers.h"
/* Debugging macros. */
#if CONFIG_DEBUG_RAM_SETUP
#define PRINTK_DEBUG(x...) printk(BIOS_DEBUG, x)
#else
#define PRINTK_DEBUG(x...)
#endif
#define RAM_COMMAND_NORMAL 0x0
#define RAM_COMMAND_NOP 0x1
#define RAM_COMMAND_PRECHARGE 0x2
#define RAM_COMMAND_MRS 0x3
#define RAM_COMMAND_CBR 0x4
#define HOSTCTRL PCI_DEV(0, 0, 2)
#define MEMCTRL PCI_DEV(0, 0, 3)
#ifdef MEM_WIDTH_32BIT_MODE
#define SDRAM1X_RA_14 30
#define SDRAM1X_RA_13 29
#define SDRAM1X_RA_12 28
#define SDRAM1X_RA_12_8bk 26
#define SDRAM1X_CA_12 15
#define SDRAM1X_CA_11 14
#define SDRAM1X_CA_09 11
#define SDRAM1X_CA_09_8bk 11
#define SDRAM1X_BA1 13
#define SDRAM1X_BA2_8bk 14
#define SDRAM1X_BA1_8bk 13
#else
#define SDRAM1X_RA_14 31
#define SDRAM1X_RA_13 30
#define SDRAM1X_RA_12 29
#define SDRAM1X_RA_12_8bk 27
#define SDRAM1X_CA_12 16
#define SDRAM1X_CA_11 15
#define SDRAM1X_CA_09 12
#define SDRAM1X_CA_09_8bk 12
#define SDRAM1X_BA1 14
#define SDRAM1X_BA2_8bk 15
#define SDRAM1X_BA1_8bk 14
#endif
#define MA_Column 0x06
#define MA_Bank 0x08
#define MA_Row 0x30
#define MA_4_Bank 0x00
#define MA_8_Bank 0x08
#define MA_12_Row 0x00
#define MA_13_Row 0x10
#define MA_14_Row 0x20
#define MA_15_Row 0x30
#define MA_9_Column 0x00
#define MA_10_Column 0x02
#define MA_11_Column 0x04
#define MA_12_Column 0x06
#define GET_SPD(i, val, tmp, reg) \
do{ \
val = 0; \
tmp = 0; \
for(i = 0; i < 2; i++) { \
if(pci_read_config8(PCI_DEV(0, 0, 4), (SCRATCH_REG_BASE + (i << 1)))) { \
tmp = get_spd_data(ctrl, i, reg); \
if(tmp > val) \
val = tmp; \
} \
} \
} while ( 0 )
#define REGISTERPRESET(bus,dev,fun,bdfspec) \
{ u8 j, reg; \
for (j=0; j<(sizeof((bdfspec))/sizeof(struct regmask)); j++) { \
printk(BIOS_DEBUG, "Writing bus " #bus " dev " #dev " fun " #fun " register "); \
printk(BIOS_DEBUG, "%02x", (bdfspec)[j].reg); \
printk(BIOS_DEBUG, "\n"); \
reg = pci_read_config8(PCI_DEV((bus), (dev), (fun)), (bdfspec)[j].reg); \
reg &= (bdfspec)[j].mask; \
reg |= (bdfspec)[j].val; \
pci_write_config8(PCI_DEV((bus), (dev), (fun)), (bdfspec)[j].reg, reg); \
} \
}
static const u8 Reg_Val[] = {
/* REG, VALUE */
0x70, 0x33,
0x71, 0x11,
0x72, 0x33,
0x73, 0x11,
0x74, 0x20,
0x75, 0x2e,
0x76, 0x64,
0x77, 0x00,
0x78, 0x44,
0x79, 0xaa,
0x7a, 0x33,
0x7b, 0xaa,
0x7c, 0x00,
0x7e, 0x33,
0x7f, 0x33,
0x80, 0x44,
0x81, 0x44,
0x82, 0x44,
0x83, 0x02,
0x50, 0x88,
0x51, 0x7b,
0x52, 0x6f,
0x53, 0x88,
0x54, 0x0e,
0x55, 0x00,
0x56, 0x00,
0x59, 0x00,
0x5d, 0x72,
0x5e, 0x88,
0x5f, 0xc7,
0x68, 0x01,
};
/* Host registers initial value */
static const u8 Host_Reg_Val[] = {
/* REG, VALUE */
0x60, 0xff,
0x61, 0xff,
0x62, 0x0f,
0x63, 0xff,
0x64, 0xff,
0x65, 0x0f,
0x66, 0xff,
0x67, 0x30,
};
static const u8 Mem_Reg_Init[] = {
/* REG, AND, OR */
0x50, 0x11, 0x66,
0x51, 0x11, 0x66,
0x52, 0x00, 0x11,
0x53, 0x00, 0x0f,
0x54, 0x00, 0x00,
0x55, 0x00, 0x00,
0x56, 0x00, 0x00,
0x57, 0x00, 0x00,
0x60, 0x00, 0x00,
0x62, 0xf7, 0x08,
0x65, 0x00, 0xd9,
0x66, 0x00, 0x80,
0x67, 0x00, 0x50, /* OR 0x00 ?? */
0x69, 0xf0, 0x00,
0x6a, 0x00, 0x00,
0x6d, 0xcf, 0xc0,
0x6e, 0xff, 0x80,
0x75, 0x0f, 0x40,
0x77, 0x00, 0x00,
0x80, 0x00, 0x00,
0x81, 0x00, 0x00,
0x82, 0x00, 0x00,
0x83, 0x00, 0x00,
0x84, 0x00, 0x00,
0x85, 0x00, 0x00,
0x86, 0xff, 0x2c, /* OR 0x28 if we don't want enable top 1M SM memory */
0x40, 0x00, 0x00,
0x7c, 0x00, 0x00,
0x7e, 0x00, 0x00,
0xa4, 0xfe, 0x00,
0xb0, 0x7f, 0x80,
0xb1, 0x00, 0xaa,
0xb4, 0xfd, 0x02,
0xb8, 0xfe, 0x00,
};
static const u8 Dram_Driving_ODT_CTRL[] = {
/* REG, VALUE */
0xd6, 0xa8,
0xd4, 0x80,
0xd0, 0x88,
0xd3, 0x01,
0xd8, 0x00,
0xda, 0x80,
};
#define Rank0_ODT 0x00
#define Rank1_ODT 0x01
#define Rank2_ODT 0x02
#define Rank3_ODT 0x03
#define NA_ODT 0x00
#define NB_ODT_75ohm 0x00
#define NB_ODT_150ohm 0x01
#define DDR2_ODT_75ohm 0x20
#define DDR2_ODT_150ohm 0x40
static const u8 ODT_TBL[] = {
/* RankMap, ODT Control Bits, DRAM & NB ODT setting */
0x01, ((NA_ODT << 6) | (NA_ODT << 4) | (NA_ODT << 2) | Rank0_ODT), (DDR2_ODT_150ohm | NB_ODT_75ohm),
0x03, ((NA_ODT << 6) | (NA_ODT << 4) | (Rank0_ODT << 2) | Rank1_ODT), (DDR2_ODT_150ohm | NB_ODT_75ohm),
0x04, ((NA_ODT << 6) | (Rank2_ODT << 4) | (NA_ODT << 2) | NA_ODT), (DDR2_ODT_150ohm | NB_ODT_75ohm),
0x05, ((NA_ODT << 6) | (Rank0_ODT << 4) | (NA_ODT << 2) | Rank2_ODT), (DDR2_ODT_75ohm | NB_ODT_150ohm),
0x07, ((NA_ODT << 6) | (Rank0_ODT << 4) | (Rank2_ODT << 2) | Rank2_ODT), (DDR2_ODT_75ohm | NB_ODT_150ohm),
0x0c, ((Rank2_ODT << 6) | (Rank3_ODT << 4) | (NA_ODT << 2) | NA_ODT), (DDR2_ODT_150ohm | NB_ODT_75ohm),
0x0d, ((Rank0_ODT << 6) | (Rank0_ODT << 4) | (NA_ODT << 2) | Rank2_ODT), (DDR2_ODT_75ohm | NB_ODT_150ohm),
0x0f, ((Rank0_ODT << 6) | (Rank0_ODT << 4) | (Rank2_ODT << 2) | Rank2_ODT), (DDR2_ODT_75ohm | NB_ODT_150ohm),
};
static const u8 DQS_DQ_TBL[] = {
/* RxE0: DRAM Timing DQS */
/* RxE2: DRAM Timing DQ */
/* RxE0, RxE2 */
0xee, 0xba,
0xee, 0xba,
0xcc, 0xba,
0xcc, 0xba,
};
static const u8 Duty_Control_DDR2[] = {
/* RxEC, RxED, RxEE, RXEF */
/* DDRII533 1~2 rank, DDRII400 */
0x84, 0x10, 0x00, 0x10,
/* DDRII533 3~4 rank */
0x44, 0x10, 0x00, 0x10,
};
static const u8 ChA_Clk_Phase_DDR2_Table[] = {
/* Rx91, Rx92, Rx93 */
/* DDRII533 1 rank */
0x04, 0x05, 0x06,
/* DDRII533 2~4 rank */
0x04, 0x05, 0x05,
/* DDRII400 */
0x02, 0x04, 0x04,
};
static const u8 DQ_DQS_Table[] = {
/* REG, VALUE */
/* DRAM DQ/DQS Output Delay Control */
0xdc, 0x65,
0xdd, 0x01,
0xde, 0xc0,
/* DRAM DQ/DQS input Capture Control */
0x78, 0x83,
0x79, 0x83,
0x7a, 0x00,
};
static const u8 DQSOChA_DDR2_Driving_Table[] = {
/* Rx70, Rx71 */
/* DDRII533 1~2 rank */
0x00, 0x01,
/* DDRII533 3~4 rank */
0x03, 0x00,
/* DDRII400 1~2 rank */
0x00, 0x04,
/* DDRII400 3~4 rank */
0x00, 0x01,
};
/************************************************************************/
/* Chipset Performance UP and other setting after DRAM Sizing Registers */
/************************************************************************/
static const u8 Dram_Table[] = {
/* REG, AND, OR */
0x60, 0xff, 0x03,
0x66, 0xcf, 0x80,
0x68, 0x00, 0x00,
0x69, 0xfd, 0x03,
0x6e, 0xff, 0x01,
0x95, 0xff, 0x40,
};
static const u8 Host_Table[] = {
/* REG, AND, OR */
0x51, 0x81, 0x7a,
0x55, 0xff, 0x06,
0x5e, 0x00, 0x88,
0x5d, 0xff, 0xb2,
};
static const u8 Init_Rank_Reg_Table[] = {
/* Rank Ending Address Registers */
0x40, 0x41, 0x42, 0x43,
/* Rank Beginning Address Registers */
0x48, 0x49, 0x4a, 0x4b,
/* Physical-to-Virtual Rank Mapping Registers */
0x54, 0x55,
};
static const u16 DDR2_MRS_table[] = {
/* CL: 2, 3, 4, 5 */
0x150, 0x1d0, 0x250, 0x2d0, /* BL=4 ;Use 1X-bandwidth MA table to init DRAM */
0x158, 0x1d8, 0x258, 0x2d8, /* BL=8 ;Use 1X-bandwidth MA table to init DRAM */
};
#define MRS_DDR2_TWR2 ((0 << 15) | (0 << 20) | (1 << 12))
#define MRS_DDR2_TWR3 ((0 << 15) | (1 << 20) | (0 << 12))
#define MRS_DDR2_TWR4 ((0 << 15) | (1 << 20) | (1 << 12))
#define MRS_DDR2_TWR5 ((1 << 15) | (0 << 20) | (0 << 12))
static const u32 DDR2_Twr_table[] = {
MRS_DDR2_TWR2,
MRS_DDR2_TWR3,
MRS_DDR2_TWR4,
MRS_DDR2_TWR5,
};
static const u8 DQSI_Rate_Table[] = {
8, /* DDRII 200 */
8, /* DDRII 266 */
8, /* DDRII 333 */
7, /* DDRII 400 */
8, /* DDRII 533 */
8, /* DDRII 666 */
};
static const u8 REFC_Table[] = {
0x65, 0x32, /* DDRII 100 */
0x86, 0x43, /* DDRII 266 */
0xa8, 0x54, /* DDRII 333 */
0xca, 0x65, /* DDRII 400 */
0xca, 0x86, /* DDRII 533 */
0xca, 0xa8, /* DDRII 666 */
};
static void do_ram_command(const struct mem_controller *ctrl, u8 command)
{
u8 reg;
reg = pci_read_config8(MEMCTRL, 0x6b);
reg &= 0xf8; /* Clear bits 2-0. */
reg |= command;
pci_write_config8(MEMCTRL, 0x6b, reg);
PRINTK_DEBUG(" Sending RAM command 0x%02x\n", reg);
}
// TODO factor out to another file
static void c7_cpu_setup(const struct mem_controller *ctrl)
{
u8 size, i;
size = sizeof(Reg_Val) / sizeof(Reg_Val[0]);
for (i = 0; i < size; i += 2)
pci_write_config8(HOSTCTRL, Reg_Val[i], Reg_Val[i + 1]);
}
static void ddr_detect(const struct mem_controller *ctrl)
{
/* FIXME: Only supports 2 ranks per DIMM */
u8 val, rsize, dimm;
u8 nrank = 0;
u8 ndimm = 0;
u8 rmap = 0;
for (dimm = 0; dimm < DIMM_SOCKETS; dimm++) {
val = get_spd_data(ctrl, dimm, 0);
if ((val == 0x80) || (val == 0xff)) {
ndimm++;
rsize = get_spd_data(ctrl, dimm, SPD_RANK_SIZE);
/* unit is 128M */
rsize = (rsize << 3) | (rsize >> 5);
val =
get_spd_data(ctrl, dimm,
SPD_MOD_ATTRIB_RANK) & SPD_MOD_ATTRIB_RANK_NUM_MASK;
switch (val) {
case 1:
pci_write_config8(PCI_DEV(0, 0, 4), (SCRATCH_RANK_1 + (dimm << 1)),
rsize);
rmap |= (1 << ((dimm << 1) + 1));
nrank++;
case 0:
pci_write_config8(PCI_DEV(0, 0, 4), (SCRATCH_RANK_0 + (dimm << 1)),
rsize);
rmap |= (1 << (dimm << 1));
nrank++;
}
}
}
pci_write_config8(PCI_DEV(0, 0, 4), SCRATCH_DIMM_NUM, ndimm);
pci_write_config8(PCI_DEV(0, 0, 4), SCRATCH_RANK_NUM, nrank);
pci_write_config8(PCI_DEV(0, 0, 4), SCRATCH_RANK_MAP, rmap);
}
static void sdram_set_safe_values(const struct mem_controller *ctrl)
{
/* The purpose of this function is to set initial values for the dram
* size and timings. It will be replaced with the SPD based function
* once the RAM commands are working with these values.
*/
u8 regs, val, t, dimm;
u32 spds, tmp;
regs = pci_read_config8(MEMCTRL, 0x6c);
if (regs & (1 << 6))
printk(BIOS_DEBUG, "DDR2 Detected.\n");
else
die("ERROR: DDR1 memory detected but not supported by coreboot.\n");
/* Enable DDR2 */
regs |= (1 << 7);
pci_write_config8(MEMCTRL, 0x6c, regs);
/* SPD 5 # of ranks */
pci_write_config8(MEMCTRL, 0x6d, 0xc0);
/**********************************************/
/* Set DRAM Freq (DDR2 533) */
/**********************************************/
/* SPD 9 SDRAM Cycle Time */
GET_SPD(dimm, spds, regs, 9);
printk(BIOS_DEBUG, "\nDDRII ");
if (spds <= 0x3d) {
printk(BIOS_DEBUG, "533");
val = DDRII_533;
t = 38;
} else if (spds <= 0x50) {
printk(BIOS_DEBUG, "400");
val = DDRII_400;
t = 50;
} else if (spds <= 0x60) {
printk(BIOS_DEBUG, "333");
val = DDRII_333;
t = 60;
} else if (spds <= 0x75) {
printk(BIOS_DEBUG, "266");
val = DDRII_266;
t = 75;
} else {
printk(BIOS_DEBUG, "200");
val = DDRII_200;
t = 100;
}
/* To store DDRII frequence */
pci_write_config8(PCI_DEV(0, 0, 4), SCRATCH_DRAM_FREQ, val);
/* Manual reset and adjust DLL when DRAM change frequency
* This is a necessary sequence.
*/
udelay(2000);
regs = pci_read_config8(MEMCTRL, 0x90);
regs |= 0x7;
pci_write_config8(MEMCTRL, 0x90, regs);
udelay(2000);
regs = pci_read_config8(MEMCTRL, 0x90);
regs &= ~0x7;
regs |= val;
pci_write_config8(MEMCTRL, 0x90, regs);
udelay(2000);
regs = pci_read_config8(MEMCTRL, 0x6b);
regs |= 0xc0;
regs &= ~0x10;
pci_write_config8(MEMCTRL, 0x6b, regs);
udelay(1);
regs |= 0x10;
pci_write_config8(MEMCTRL, 0x6b, regs);
udelay(1);
regs &= ~0xc0;
pci_write_config8(MEMCTRL, 0x6b, regs);
regs = pci_read_config8(MEMCTRL, 0x6f);
regs |= 0x1;
pci_write_config8(MEMCTRL, 0x6f, regs);
/**********************************************/
/* Set DRAM Timing Setting (DDR2 533) */
/**********************************************/
/* SPD 9 18 23 25 CAS Latency NB3DRAM_REG62[2:0] */
/* Read SPD byte 18 CAS Latency */
GET_SPD(dimm, spds, regs, SPD_CAS_LAT);
printk(BIOS_DEBUG, "\nCAS Supported ");
if (spds & SPD_CAS_LAT_2)
printk(BIOS_DEBUG, "2 ");
if (spds & SPD_CAS_LAT_3)
printk(BIOS_DEBUG, "3 ");
if (spds & SPD_CAS_LAT_4)
printk(BIOS_DEBUG, "4 ");
if (spds & SPD_CAS_LAT_5)
printk(BIOS_DEBUG, "5 ");
if (spds & SPD_CAS_LAT_6)
printk(BIOS_DEBUG, "6");
/* We don't consider CAS = 6, because CX700 doesn't support it */
printk(BIOS_DEBUG, "\n CAS:");
if (spds & SPD_CAS_LAT_5) {
printk(BIOS_DEBUG, "Starting at CL5");
val = 0x3;
/* See whether we can improve it */
GET_SPD(dimm, tmp, regs, SPD_CAS_LAT_MIN_X_1);
if ((spds & SPD_CAS_LAT_4) && (tmp < 0x50)) {
printk(BIOS_DEBUG, "\n... going to CL4");
val = 0x2;
}
GET_SPD(dimm, tmp, regs, SPD_CAS_LAT_MIN_X_2);
if ((spds & SPD_CAS_LAT_3) && (tmp < 0x50)) {
printk(BIOS_DEBUG, "\n... going to CL3");
val = 0x1;
}
} else {
printk(BIOS_DEBUG, "Starting at CL4");
val = 0x2;
GET_SPD(dimm, tmp, regs, SPD_CAS_LAT_MIN_X_1);
if ((spds & SPD_CAS_LAT_3) && (tmp < 0x50)) {
printk(BIOS_DEBUG, "\n... going to CL3");
val = 0x1;
}
GET_SPD(dimm, tmp, regs, SPD_CAS_LAT_MIN_X_2);
if ((spds & SPD_CAS_LAT_2) && (tmp < 0x50)) {
printk(BIOS_DEBUG, "\n... going to CL2");
val = 0x0;
}
}
regs = pci_read_config8(MEMCTRL, 0x62);
regs &= ~0x7;
regs |= val;
pci_write_config8(MEMCTRL, 0x62, regs);
/* SPD 27 Trp NB3DRAM_REG64[3:2] */
GET_SPD(dimm, spds, regs, SPD_TRP);
printk(BIOS_DEBUG, "\nTrp %d", spds);
spds >>= 2;
for (val = 2; val <= 5; val++) {
if (spds <= (val * t / 10)) {
val = val - 2;
break;
}
}
val <<= 2;
regs = pci_read_config8(MEMCTRL, 0x64);
regs &= ~0xc;
regs |= val;
pci_write_config8(MEMCTRL, 0x64, regs);
/* SPD 29 Trcd NB3DRAM_REG64[7:6] */
GET_SPD(dimm, spds, regs, SPD_TRCD);
printk(BIOS_DEBUG, "\nTrcd %d", spds);
spds >>= 2;
for (val = 2; val <= 5; val++) {
if (spds <= (val * t / 10)) {
val = val - 2;
break;
}
}
val <<= 6;
regs = pci_read_config8(MEMCTRL, 0x64);
regs &= ~0xc0;
regs |= val;
pci_write_config8(MEMCTRL, 0x64, regs);
/* SPD 30 Tras NB3DRAM_REG62[7:4] */
GET_SPD(dimm, spds, regs, SPD_TRAS);
printk(BIOS_DEBUG, "\nTras %d", spds);
for (val = 5; val <= 20; val++) {
if (spds <= (val * t / 10)) {
val = val - 5;
break;
}
}
val <<= 4;
regs = pci_read_config8(MEMCTRL, 0x62);
regs &= ~0xf0;
regs |= val;
pci_write_config8(MEMCTRL, 0x62, regs);
/* SPD 42 SPD 40 Trfc NB3DRAM_REG61[5:0] */
GET_SPD(dimm, spds, regs, SPD_TRFC);
printk(BIOS_DEBUG, "\nTrfc %d", spds);
tmp = spds;
GET_SPD(dimm, spds, regs, SPD_EX_TRC_TRFC);
if (spds & 0x1)
tmp += 256;
if (spds & 0xe)
tmp++;
for (val = 8; val <= 71; val++) {
if (tmp <= (val * t / 10)) {
val = val - 8;
break;
}
}
regs = pci_read_config8(MEMCTRL, 0x61);
regs &= ~0x3f;
regs |= val;
pci_write_config8(MEMCTRL, 0x61, regs);
/* SPD 28 Trrd NB3DRAM_REG63[7:6] */
GET_SPD(dimm, spds, regs, SPD_TRRD);
for (val = 2; val <= 5; val++) {
if (spds <= (val * t / 10)) {
val = val - 2;
break;
}
}
val <<= 6;
printk(BIOS_DEBUG, "\nTrrd val = 0x%x", val);
regs = pci_read_config8(MEMCTRL, 0x63);
regs &= ~0xc0;
regs |= val;
pci_write_config8(MEMCTRL, 0x63, regs);
/* SPD 36 Twr NB3DRAM_REG61[7:6] */
GET_SPD(dimm, spds, regs, SPD_TWR);
for (val = 2; val <= 5; val++) {
if (spds <= (val * t / 10)) {
val = val - 2;
break;
}
}
val <<= 6;
printk(BIOS_DEBUG, "\nTwr val = 0x%x", val);
regs = pci_read_config8(MEMCTRL, 0x61);
regs &= ~0xc0;
regs |= val;
pci_write_config8(MEMCTRL, 0x61, regs);
/* SPD 37 Twtr NB3DRAM_REG63[1] */
GET_SPD(dimm, spds, regs, SPD_TWTR);
spds >>= 2;
printk(BIOS_DEBUG, "\nTwtr 0x%x", spds);
if (spds <= (t * 2 / 10))
val = 0;
else
val = 1;
val <<= 1;
printk(BIOS_DEBUG, "\nTwtr val = 0x%x", val);
regs = pci_read_config8(MEMCTRL, 0x63);
regs &= ~0x2;
regs |= val;
pci_write_config8(MEMCTRL, 0x63, regs);
/* SPD 38 Trtp NB3DRAM_REG63[3] */
GET_SPD(dimm, spds, regs, SPD_TRTP);
spds >>= 2;
printk(BIOS_DEBUG, "\nTrtp 0x%x", spds);
if (spds <= (t * 2 / 10))
val = 0;
else
val = 1;
val <<= 3;
printk(BIOS_DEBUG, "\nTrtp val = 0x%x", val);
regs = pci_read_config8(MEMCTRL, 0x63);
regs &= ~0x8;
regs |= val;
pci_write_config8(MEMCTRL, 0x63, regs);
/**********************************************/
/* Set DRAM DRDY Setting */
/**********************************************/
/* Write slowest value to register */
tmp = sizeof(Host_Reg_Val) / sizeof(Host_Reg_Val[0]);
for (val = 0; val < tmp; val += 2)
pci_write_config8(HOSTCTRL, Host_Reg_Val[val], Host_Reg_Val[val + 1]);
/* F2_RX51[7]=0, disable DRDY timing */
regs = pci_read_config8(HOSTCTRL, 0x51);
regs &= ~0x80;
pci_write_config8(HOSTCTRL, 0x51, regs);
/**********************************************/
/* Set DRAM BurstLength */
/**********************************************/
regs = pci_read_config8(MEMCTRL, 0x6c);
for (dimm = 0; dimm < DIMM_SOCKETS; dimm++) {
if (pci_read_config8(PCI_DEV(0, 0, 4), (SCRATCH_REG_BASE + (dimm << 1)))) {
spds = get_spd_data(ctrl, dimm, 16);
if (!(spds & 0x8))
break;
}
}
if (dimm == 2)
regs |= 0x8;
pci_write_config8(MEMCTRL, 0x6c, regs);
val = pci_read_config8(HOSTCTRL, 0x54);
val &= ~0x10;
if (dimm == 2)
val |= 0x10;
pci_write_config8(HOSTCTRL, 0x54, val);
/**********************************************/
/* Set DRAM Driving Setting */
/**********************************************/
/* DRAM Timing ODT */
tmp = sizeof(Dram_Driving_ODT_CTRL) / sizeof(Dram_Driving_ODT_CTRL[0]);
for (val = 0; val < tmp; val += 2)
pci_write_config8(MEMCTRL, Dram_Driving_ODT_CTRL[val],
Dram_Driving_ODT_CTRL[val + 1]);
regs = pci_read_config8(PCI_DEV(0, 0, 4), SCRATCH_RANK_NUM);
val = pci_read_config8(MEMCTRL, 0xd5);
val &= ~0xaa;
switch (regs) {
case 3:
case 2:
val |= 0xa0;
break;
default:
val |= 0x80;
}
regs = pci_read_config8(PCI_DEV(0, 0, 4), SCRATCH_DIMM_NUM);
if (regs == 1)
val |= 0xa;
pci_write_config8(MEMCTRL, 0xd5, val);
regs = pci_read_config8(PCI_DEV(0, 0, 4), SCRATCH_DIMM_NUM);
val = pci_read_config8(MEMCTRL, 0xd6);
val &= ~0x2;
if (regs == 1)
val |= 0x2;
pci_write_config8(MEMCTRL, 0xd6, val);
regs = pci_read_config8(PCI_DEV(0, 0, 4), SCRATCH_RANK_MAP);
tmp = sizeof(ODT_TBL) / sizeof(ODT_TBL[0]);
for (val = 0; val < tmp; val += 3) {
if (regs == ODT_TBL[val]) {
pci_write_config8(MEMCTRL, 0xd8, ODT_TBL[val + 1]);
/* Store DRAM & NB ODT setting in d0f4_Rxd8 */
pci_write_config8(PCI_DEV(0, 0, 4), SCRATCH_DRAM_NB_ODT, ODT_TBL[val + 2]);
break;
}
}
pci_write_config8(MEMCTRL, 0xd9, 0x0a);
regs = pci_read_config8(PCI_DEV(0, 0, 4), SCRATCH_RANK_NUM);
regs--;
regs = regs << 1;
pci_write_config8(MEMCTRL, 0xe0, DQS_DQ_TBL[regs++]);
pci_write_config8(MEMCTRL, 0xe2, DQS_DQ_TBL[regs]);
/* DRAM Timing CS */
pci_write_config8(MEMCTRL, 0xe4, 0x66);
/* DRAM Timing MAA */
val = 0;
for (dimm = 0; dimm < DIMM_SOCKETS; dimm++) {
if (pci_read_config8(PCI_DEV(0, 0, 4), (SCRATCH_REG_BASE + (dimm << 1)))) {
spds = get_spd_data(ctrl, dimm, SPD_PRI_WIDTH);
spds = 64 / spds;
if (pci_read_config8
(PCI_DEV(0, 0, 4), (SCRATCH_REG_BASE + (dimm << 1) + 1)))
spds = spds << 1;
val += spds;
}
}
printk(BIOS_DEBUG, "\nchip #%d", val);
if (val > 18)
regs = 0xdb;
else
regs = 0x86;
pci_write_config8(MEMCTRL, 0xe8, regs);
/* DRAM Timing MAB */
pci_write_config8(MEMCTRL, 0xe9, 0x0);
/* DRAM Timing DCLK VT8454C always 0x66 */
pci_write_config8(MEMCTRL, 0xe6, 0xaa);
/**********************************************/
/* Set DRAM Duty Control */
/**********************************************/
regs = pci_read_config8(PCI_DEV(0, 0, 4), SCRATCH_RANK_NUM);
switch (regs) {
case 1:
case 2: /* 1~2 rank */
val = 0;
break;
case 3:
case 4: /* 3~4 rank */
regs = pci_read_config8(PCI_DEV(0, 0, 4), SCRATCH_DRAM_FREQ);
if (regs == DDRII_533)
val = 4;
else /* DDRII-400 */
val = 0;
break;
}
regs = 0xec;
for (t = 0; t < 4; t++) {
pci_write_config8(MEMCTRL, regs, Duty_Control_DDR2[val]);
regs++;
val++;
}
/**********************************************/
/* Set DRAM Clock Control */
/**********************************************/
/* Write Data Phase */
val = pci_read_config8(PCI_DEV(0, 0, 4), SCRATCH_DRAM_FREQ);
regs = pci_read_config8(MEMCTRL, 0x75);
regs &= 0xf0;
switch (val) {
case DDRII_533:
pci_write_config8(MEMCTRL, 0x74, 0x07);
regs |= 0x7;
break;
case DDRII_400:
default:
pci_write_config8(MEMCTRL, 0x74, 0x05);
regs |= 0x5;
break;
}
pci_write_config8(MEMCTRL, 0x75, regs);
pci_write_config8(MEMCTRL, 0x76, 0x80);
/* Clock Phase Control for FeedBack Mode */
regs = pci_read_config8(MEMCTRL, 0x90);
// regs |= 0x80;
pci_write_config8(MEMCTRL, 0x90, regs);
regs = pci_read_config8(PCI_DEV(0, 0, 4), SCRATCH_DRAM_FREQ);
switch (regs) {
case DDRII_533:
regs = pci_read_config8(PCI_DEV(0, 0, 4), SCRATCH_RANK_NUM);
if (regs == 1)
val = 0;
else
val = 3;
break;
case DDRII_400:
default:
val = 6;
break;
}
regs = pci_read_config8(MEMCTRL, 0x91);
regs &= ~0xc0;
regs |= 0x80;
pci_write_config8(MEMCTRL, 0x91, regs);
regs = 0x91;
for (t = 0; t < 3; t++) {
dimm = pci_read_config8(MEMCTRL, regs);
dimm &= ~0x7;
dimm |= ChA_Clk_Phase_DDR2_Table[val];
pci_write_config8(MEMCTRL, regs, dimm);
regs++;
val++;
}
pci_write_config8(MEMCTRL, 0x97, 0x12);
pci_write_config8(MEMCTRL, 0x98, 0x33);
regs = pci_read_config8(PCI_DEV(0, 0, 4), SCRATCH_RANK_0);
val = pci_read_config8(PCI_DEV(0, 0, 4), SCRATCH_RANK_2);
if (regs && val)
pci_write_config8(MEMCTRL, 0x9d, 0x00);
else
pci_write_config8(MEMCTRL, 0x9d, 0x0f);
tmp = sizeof(DQ_DQS_Table) / sizeof(DQ_DQS_Table[0]);
for (val = 0; val < tmp; val += 2)
pci_write_config8(MEMCTRL, DQ_DQS_Table[val], DQ_DQS_Table[val + 1]);
regs = pci_read_config8(PCI_DEV(0, 0, 4), SCRATCH_DRAM_FREQ);
if (regs == DDRII_533)
pci_write_config8(MEMCTRL, 0x7b, 0xa0);
else
pci_write_config8(MEMCTRL, 0x7b, 0x10);
/***************************************************/
/* Set necessary register before DRAM initialize */
/***************************************************/
tmp = sizeof(Mem_Reg_Init) / sizeof(Mem_Reg_Init[0]);
for (val = 0; val < tmp; val += 3) {
regs = pci_read_config8(MEMCTRL, Mem_Reg_Init[val]);
regs &= Mem_Reg_Init[val + 1];
regs |= Mem_Reg_Init[val + 2];
pci_write_config8(MEMCTRL, Mem_Reg_Init[val], regs);
}
regs = pci_read_config8(HOSTCTRL, 0x51);
regs &= 0xbf; // Clear bit 6 Disable Read Around Write
pci_write_config8(HOSTCTRL, 0x51, regs);
regs = pci_read_config8(HOSTCTRL, 0x54);
t = regs >> 5;
val = pci_read_config8(HOSTCTRL, 0x57);
dimm = val >> 5;
if (t == dimm)
t = 0x0;
else
t = 0x1;
regs &= ~0x1;
regs |= t;
val &= ~0x1;
val |= t;
pci_write_config8(HOSTCTRL, 0x57, val);
regs = pci_read_config8(HOSTCTRL, 0x51);
regs |= t;
pci_write_config8(HOSTCTRL, 0x51, regs);
regs = pci_read_config8(MEMCTRL, 0x90);
regs &= 0x7;
val = 0;
if (regs < 0x2)
val = 0x80;
regs = pci_read_config8(MEMCTRL, 0x76);
regs &= 0x80;
regs |= val;
pci_write_config8(MEMCTRL, 0x76, regs);
regs = pci_read_config8(MEMCTRL, 0x6f);
regs |= 0x10;
pci_write_config8(MEMCTRL, 0x6f, regs);
/***************************************************/
/* Find suitable DQS value for ChA and ChB */
/***************************************************/
// Set DQS output delay for Channel A
regs = pci_read_config8(PCI_DEV(0, 0, 4), SCRATCH_DRAM_FREQ);
val = pci_read_config8(PCI_DEV(0, 0, 4), SCRATCH_RANK_NUM);
switch (regs) {
case DDRII_533:
if (val < 2)
val = 0;
else
val = 2;
break;
case DDRII_400:
default:
if (val < 2)
val = 4;
else
val = 6;
break;
}
for (t = 0; t < 2; t++)
pci_write_config8(MEMCTRL, (0x70 + t), DQSOChA_DDR2_Driving_Table[val + t]);
// Set DQS output delay for Channel B
pci_write_config8(MEMCTRL, 0x72, 0x0);
regs = pci_read_config8(PCI_DEV(0, 0, 4), SCRATCH_RANK_0);
val = pci_read_config8(PCI_DEV(0, 0, 4), SCRATCH_RANK_2);
if (regs && val)
pci_write_config8(MEMCTRL, 0x73, 0xfd);
else
pci_write_config8(MEMCTRL, 0x73, 0x01);
}
static void sdram_set_registers(const struct mem_controller *ctrl)
{
c7_cpu_setup(ctrl);
ddr_detect(ctrl);
sdram_set_safe_values(ctrl);
}
static void step_20_21(const struct mem_controller *ctrl)
{
u8 val;
// Step 20
udelay(200);
val = pci_read_config8(PCI_DEV(0, 0, 4), SCRATCH_DRAM_NB_ODT);
if (val & DDR2_ODT_150ohm)
read32((void *)0x102200);
else
read32((void *)0x102020);
/* Step 21. Normal operation */
printk(BIOS_SPEW, "RAM Enable 5: Normal operation\n");
do_ram_command(ctrl, RAM_COMMAND_NORMAL);
udelay(3);
}
static void step_2_19(const struct mem_controller *ctrl)
{
u32 i;
u8 val;
// Step 2
val = pci_read_config8(MEMCTRL, 0x69);
val &= ~0x03;
pci_write_config8(MEMCTRL, 0x69, val);
/* Step 3 Apply NOP. */
printk(BIOS_SPEW, "RAM Enable 1: Apply NOP\n");
do_ram_command(ctrl, RAM_COMMAND_NOP);
udelay(15);
// Step 4
printk(BIOS_SPEW, "SEND: ");
read32(zeroptr);
printk(BIOS_SPEW, "OK\n");
// Step 5
udelay(400);
/* 6. Precharge all. Wait tRP. */
printk(BIOS_SPEW, "RAM Enable 2: Precharge all\n");
do_ram_command(ctrl, RAM_COMMAND_PRECHARGE);
// Step 7
printk(BIOS_SPEW, "SEND: ");
read32(zeroptr);
printk(BIOS_SPEW, "OK\n");
/* Step 8. Mode register set. */
printk(BIOS_SPEW, "RAM Enable 4: Mode register set\n");
do_ram_command(ctrl, RAM_COMMAND_MRS); //enable dll
// Step 9
printk(BIOS_SPEW, "SEND: ");
val = pci_read_config8(PCI_DEV(0, 0, 4), SCRATCH_DRAM_NB_ODT);
if (val & DDR2_ODT_150ohm)
read32((void *)0x102200); //DDR2_ODT_150ohm
else
read32((void *)0x102020);
printk(BIOS_SPEW, "OK\n");
// Step 10
printk(BIOS_SPEW, "SEND: ");
read32((void *)0x800);
printk(BIOS_SPEW, "OK\n");
/* Step 11. Precharge all. Wait tRP. */
printk(BIOS_SPEW, "RAM Enable 2: Precharge all\n");
do_ram_command(ctrl, RAM_COMMAND_PRECHARGE);
// Step 12
printk(BIOS_SPEW, "SEND: ");
read32(zeroptr);
printk(BIOS_SPEW, "OK\n");
/* Step 13. Perform 8 refresh cycles. Wait tRC each time. */
printk(BIOS_SPEW, "RAM Enable 3: CBR\n");
do_ram_command(ctrl, RAM_COMMAND_CBR);
/* JEDEC says only twice, do 8 times for posterity */
// Step 16: Repeat Step 14 and 15 another 7 times
for (i = 0; i < 8; i++) {
// Step 14
read32(zeroptr);
printk(BIOS_SPEW, ".");
// Step 15
udelay(100);
}
/* Step 17. Mode register set. Wait 200us. */
printk(BIOS_SPEW, "\nRAM Enable 4: Mode register set\n");
//safe value for now, BL=8, WR=4, CAS=4
do_ram_command(ctrl, RAM_COMMAND_MRS);
udelay(200);
/* Use Single Chanel temporarily */
val = pci_read_config8(MEMCTRL, 0x6c);
if (val & 0x8) { /* Burst Length = 8 */
val = pci_read_config8(MEMCTRL, 0x62);
val &= 0x7;
i = DDR2_MRS_table[4 + val];
} else {
val = pci_read_config8(MEMCTRL, 0x62);
val &= 0x7;
i = DDR2_MRS_table[val];
}
// Step 18
val = pci_read_config8(MEMCTRL, 0x61);
val = val >> 6;
i |= DDR2_Twr_table[val];
read32((void *)i);
printk(BIOS_DEBUG, "MRS = %08x\n", i);
udelay(15);
// Step 19
val = pci_read_config8(PCI_DEV(0, 0, 4), SCRATCH_DRAM_NB_ODT);
if (val & DDR2_ODT_150ohm)
read32((void *)0x103e00); //EMRS OCD Default
else
read32((void *)0x103c20);
}
static void sdram_set_vr(const struct mem_controller *ctrl, u8 num)
{
u8 reg, val;
val = 0x54 + (num >> 1);
reg = pci_read_config8(MEMCTRL, val);
reg &= (0xf << (4 * (num & 0x1)));
reg |= (((0x8 | num) << 4) >> (4 * (num & 0x1)));
pci_write_config8(MEMCTRL, val, reg);
}
static void sdram_ending_addr(const struct mem_controller *ctrl, u8 num)
{
u8 reg, val;
/* Set Ending Address */
val = 0x40 + num;
reg = pci_read_config8(MEMCTRL, val);
reg += 0x10;
pci_write_config8(MEMCTRL, val, reg);
/* Set Beginning Address */
val = 0x48 + num;
pci_write_config8(MEMCTRL, val, 0x0);
}
static void sdram_clear_vr_addr(const struct mem_controller *ctrl, u8 num)
{
u8 reg, val;
val = 0x54 + (num >> 1);
reg = pci_read_config8(MEMCTRL, val);
reg = ~(0x80 >> (4 * (num & 0x1)));
pci_write_config8(MEMCTRL, val, reg);
val = 0x40 + num;
reg = pci_read_config8(MEMCTRL, val);
reg -= 0x10;
pci_write_config8(MEMCTRL, val, reg);
val = 0x48 + num;
pci_write_config8(MEMCTRL, val, 0x0);
}
/* Perform sizing DRAM by dynamic method */
static void sdram_calc_size(const struct mem_controller *ctrl, u8 num)
{
u8 ca, ra, ba, reg;
ba = pci_read_config8(PCI_DEV(0, 0, 4), SCRATCH_FLAGS);
if (ba == 8) {
write8(zeroptr, 0x0d);
ra = read8(zeroptr);
write8((void *)(1 << SDRAM1X_RA_12_8bk), 0x0c);
ra = read8(zeroptr);
write8(zeroptr, 0x0a);
ca = read8(zeroptr);
write8((void *)(1 << SDRAM1X_CA_09_8bk), 0x0c);
ca = read8(zeroptr);
write8(zeroptr, 0x03);
ba = read8(zeroptr);
write8((void *)(1 << SDRAM1X_BA2_8bk), 0x02);
ba = read8(zeroptr);
write8((void *)(1 << SDRAM1X_BA1_8bk), 0x01);
ba = read8(zeroptr);
} else {
write8(zeroptr, 0x0f);
ra = read8(zeroptr);
write8((void *)(1 << SDRAM1X_RA_14), 0x0e);
ra = read8(zeroptr);
write8((void *)(1 << SDRAM1X_RA_13), 0x0d);
ra = read8(zeroptr);
write8((void *)(1 << SDRAM1X_RA_12), 0x0c);
ra = read8(zeroptr);
write8(zeroptr, 0x0c);
ca = read8(zeroptr);
write8((void *)(1 << SDRAM1X_CA_12), 0x0b);
ca = read8(zeroptr);
write8((void *)(1 << SDRAM1X_CA_11), 0x0a);
ca = read8(zeroptr);
write8((void *)(1 << SDRAM1X_CA_09), 0x09);
ca = read8(zeroptr);
write8(zeroptr, 0x02);
ba = read8(zeroptr);
write8((void *)(1 << SDRAM1X_BA1), 0x01);
ba = read8(zeroptr);
}
if (ra < 10 || ra > 15)
die("bad RA");
if (ca < 8 || ca > 12)
die("bad CA");
if (ba < 1 || ba > 3)
die("bad BA");
/* Calculate MA type save to scratch register */
reg = 0;
switch (ra) {
case 12:
reg |= MA_12_Row;
break;
case 13:
reg |= MA_13_Row;
break;
case 14:
reg |= MA_14_Row;
break;
default:
reg |= MA_15_Row;
}
switch (ca) {
case 9:
reg |= MA_9_Column;
break;
case 10:
reg |= MA_10_Column;
break;
case 11:
reg |= MA_11_Column;
break;
default:
reg |= MA_12_Column;
}
switch (ba) {
case 3:
reg |= MA_8_Bank;
break;
default:
reg |= MA_4_Bank;
}
pci_write_config8(PCI_DEV(0, 0, 4), (SCRATCH_RANK0_MA_REG + num), reg);
if (ra >= 13)
pci_write_config8(PCI_DEV(0, 0, 4), SCRATCH_DRAM_256M_BIT, 1);
/* Calculate rank size save to scratch register */
ra = ra + ca + ba + 3 - 26; /* 1 unit = 64M */
ra = 1 << ra;
pci_write_config8(PCI_DEV(0, 0, 4), (SCRATCH_RANK0_SIZE_REG + num), ra);
}
static void sdram_enable(const struct mem_controller *ctrl)
{
u8 reg8;
u8 val, i;
device_t dev;
u8 dl, dh;
u32 quot;
/* Init Present Bank */
val = sizeof(Init_Rank_Reg_Table) / sizeof(Init_Rank_Reg_Table[0]);
for (i = 0; i < val; i++)
pci_write_config8(MEMCTRL, Init_Rank_Reg_Table[i], 0x0);
/* Init other banks */
for (i = 0; i < 4; i++) {
reg8 = pci_read_config8(PCI_DEV(0, 0, 4), (SCRATCH_RANK_0 + i));
if (reg8) {
sdram_set_vr(ctrl, i);
sdram_ending_addr(ctrl, i);
step_2_19(ctrl);
step_20_21(ctrl);
sdram_clear_vr_addr(ctrl, i);
}
}
if (IS_ENABLED(MEM_WIDTH_32BIT_MODE)) {
/********************************************************/
/* Set Dram 32bit Mode */
/********************************************************/
reg8 = pci_read_config8(MEMCTRL, 0x6c);
reg8 |= 0x20;
pci_write_config8(MEMCTRL, 0x6c, reg8);
}
/****************************************************************/
/* Find the DQSI Low/High bound and save it to Scratch register */
/****************************************************************/
for (dl = 0; dl < 0x3f; dl += 2) {
reg8 = dl & 0x3f;
reg8 |= 0x80; /* Set Manual Mode */
pci_write_config8(MEMCTRL, 0x77, reg8);
for (i = 0; i < 4; i++) {
reg8 = pci_read_config8(PCI_DEV(0, 0, 4), (SCRATCH_RANK_0 + i));
if (reg8) {
sdram_set_vr(ctrl, i);
sdram_ending_addr(ctrl, i);
write32(zeroptr, 0x55555555);
write32((void *)4, 0x55555555);
udelay(15);
if (read32(zeroptr) != 0x55555555)
break;
if (read32((void *)4) != 0x55555555)
break;
write32(zeroptr, 0xaaaaaaaa);
write32((void *)4, 0xaaaaaaaa);
udelay(15);
if (read32(zeroptr) != 0xaaaaaaaa)
break;
if (read32((void *)4) != 0xaaaaaaaa)
break;
sdram_clear_vr_addr(ctrl, i);
}
}
if (i == 4)
break;
else
sdram_clear_vr_addr(ctrl, i);
}
printk(BIOS_DEBUG, "\nDQSI Low %08x", dl);
for (dh = dl; dh < 0x3f; dh += 2) {
reg8 = dh & 0x3f;
reg8 |= 0x80; /* Set Manual Mode */
pci_write_config8(MEMCTRL, 0x77, reg8);
for (i = 0; i < 4; i++) {
reg8 = pci_read_config8(PCI_DEV(0, 0, 4), (SCRATCH_RANK_0 + i));
if (reg8) {
sdram_set_vr(ctrl, i);
sdram_ending_addr(ctrl, i);
write32(zeroptr, 0x55555555);
write32((void *)4, 0x55555555);
udelay(15);
if (read32(zeroptr) != 0x55555555)
break;
if (read32((void *)4) != 0x55555555)
break;
write32(zeroptr, 0xaaaaaaaa);
write32((void *)4, 0xaaaaaaaa);
udelay(15);
if (read32(zeroptr) != 0xaaaaaaaa)
break;
if (read32((void *)4) != 0xaaaaaaaa)
break;
sdram_clear_vr_addr(ctrl, i);
}
}
if (i != 4) {
sdram_clear_vr_addr(ctrl, i);
break;
}
}
printk(BIOS_DEBUG, "\nDQSI High %02x", dh);
pci_write_config8(PCI_DEV(0, 0, 4), SCRATCH_CHA_DQSI_LOW_REG, dl);
pci_write_config8(PCI_DEV(0, 0, 4), SCRATCH_CHA_DQSI_HIGH_REG, dh);
reg8 = pci_read_config8(MEMCTRL, 0X90) & 0X7;
val = DQSI_Rate_Table[reg8];
quot = dh - dl;
quot = quot * val;
quot >>= 4;
val = quot + dl;
pci_write_config8(PCI_DEV(0, 0, 4), SCRATCH_ChA_DQSI_REG, val);
reg8 = val & 0x3f;
reg8 |= 0x80;
pci_write_config8(MEMCTRL, 0x77, reg8);
/****************************************************************/
/* Find out the lowest Bank Interleave and Set Register */
/****************************************************************/
#if 0
//TODO
reg8 = pci_read_config8(MEMCTRL, 0x69);
reg8 &= ~0xc0;
reg8 |= 0x80; //8 banks
pci_write_config8(MEMCTRL, 0x69, reg8);
#endif
dl = 2;
for (i = 0; i < 4; i++) {
reg8 = pci_read_config8(PCI_DEV(0, 0, 4), (SCRATCH_RANK_0 + i));
if (reg8) {
reg8 = get_spd_data(ctrl, (i >> 1), 17);
sdram_set_vr(ctrl, i);
sdram_ending_addr(ctrl, i);
if (reg8 == 4) {
write8(zeroptr, 0x02);
val = read8(zeroptr);
write8((void *)(1 << SDRAM1X_BA1), 0x01);
val = read8(zeroptr);
} else {
write8(zeroptr, 0x03);
val = read8(zeroptr);
write8((void *)(1 << SDRAM1X_BA2_8bk), 0x02);
val = read8(zeroptr);
write8((void *)(1 << SDRAM1X_BA1_8bk), 0x01);
val = read8(zeroptr);
}
if (val < dl)
dl = val;
sdram_clear_vr_addr(ctrl, i);
}
}
dl <<= 6;
reg8 = pci_read_config8(MEMCTRL, 0x69);
reg8 &= ~0xc0;
reg8 |= dl;
pci_write_config8(MEMCTRL, 0x69, reg8);
/****************************************************************/
/* DRAM Sizing and Fill MA type */
/****************************************************************/
for (i = 0; i < 4; i++) {
val = pci_read_config8(PCI_DEV(0, 0, 4), (SCRATCH_RANK_0 + i));
if (val) {
reg8 = get_spd_data(ctrl, (i >> 1), 17);
pci_write_config8(PCI_DEV(0, 0, 4), SCRATCH_FLAGS, reg8);
if (reg8 == 4) {
/* Use MA Type 3 for DRAM sizing */
reg8 = pci_read_config8(MEMCTRL, 0x50);
reg8 &= 0x11;
reg8 |= 0x66;
pci_write_config8(MEMCTRL, 0x50, reg8);
pci_write_config8(MEMCTRL, 0x51, reg8);
} else {
/* Use MA Type 5 for DRAM sizing */
reg8 = pci_read_config8(MEMCTRL, 0x50);
reg8 &= 0x11;
reg8 |= 0xaa;
pci_write_config8(MEMCTRL, 0x50, reg8);
pci_write_config8(MEMCTRL, 0x51, reg8);
reg8 = pci_read_config8(MEMCTRL, 0x53);
reg8 &= 0x0f;
reg8 |= 0x90;
pci_write_config8(MEMCTRL, 0x53, reg8);
}
sdram_set_vr(ctrl, i);
val = 0x40 + i;
reg8 = pci_read_config8(MEMCTRL, val);
/* max size 3G for new MA table */
reg8 += 0x30;
pci_write_config8(MEMCTRL, val, reg8);
/* Set Beginning Address */
val = 0x48 + i;
pci_write_config8(MEMCTRL, val, 0x0);
sdram_calc_size(ctrl, i);
/* Clear */
val = 0x54 + (i >> 1);
reg8 = pci_read_config8(MEMCTRL, val);
reg8 = ~(0x80 >> (4 * (i & 0x1)));
pci_write_config8(MEMCTRL, val, reg8);
val = 0x40 + i;
reg8 = pci_read_config8(MEMCTRL, val);
reg8 -= 0x30;
pci_write_config8(MEMCTRL, val, reg8);
val = 0x48 + i;
pci_write_config8(MEMCTRL, val, 0x0);
}
}
/* Clear MA Type */
reg8 = pci_read_config8(MEMCTRL, 0x50);
reg8 &= 0x11;
pci_write_config8(MEMCTRL, 0x50, reg8);
pci_write_config8(MEMCTRL, 0x51, reg8);
reg8 = pci_read_config8(MEMCTRL, 0x6b);
reg8 &= ~0x08;
pci_write_config8(MEMCTRL, 0x6b, reg8);
/****************************************************************/
/* DRAM re-initialize for burst length */
/****************************************************************/
for (i = 0; i < 4; i++) {
reg8 = pci_read_config8(PCI_DEV(0, 0, 4), (SCRATCH_RANK_0 + i));
if (reg8) {
sdram_set_vr(ctrl, i);
sdram_ending_addr(ctrl, i);
step_2_19(ctrl);
step_20_21(ctrl);
sdram_clear_vr_addr(ctrl, i);
}
}
/****************************************************************/
/* Set the MA Type */
/****************************************************************/
reg8 = pci_read_config8(MEMCTRL, 0x50);
reg8 &= 0x11;
pci_write_config8(MEMCTRL, 0x50, reg8);
reg8 = pci_read_config8(MEMCTRL, 0x51);
reg8 &= 0x11;
pci_write_config8(MEMCTRL, 0x51, reg8);
reg8 = pci_read_config8(MEMCTRL, 0x6b);
reg8 &= ~0x08;
pci_write_config8(MEMCTRL, 0x6b, reg8);
for (i = 0; i < 4; i += 2) {
reg8 = pci_read_config8(PCI_DEV(0, 0, 4), (SCRATCH_RANK_0 + i));
if (reg8) {
reg8 = pci_read_config8(PCI_DEV(0, 0, 4), (SCRATCH_RANK0_MA_REG + i));
reg8 &= (MA_Bank + MA_Column);
val = pci_read_config8(MEMCTRL, 0x50);
if (i == 0) {
reg8 <<= 4;
val &= 0x1f;
} else
val &= 0xf1;
val |= reg8;
pci_write_config8(MEMCTRL, 0x50, val);
}
}
/****************************************************************/
/* Set Start and Ending Address */
/****************************************************************/
dl = 0; /* Begin Address */
dh = 0; /* Ending Address */
for (i = 0; i < 4; i++) {
reg8 = pci_read_config8(PCI_DEV(0, 0, 4), (SCRATCH_RANK_0 + i));
if (reg8) {
reg8 = pci_read_config8(PCI_DEV(0, 0, 4), (SCRATCH_RANK0_SIZE_REG + i));
if (reg8 == 0)
continue;
dh += reg8;
pci_write_config8(MEMCTRL, (0x40 + i), dh);
pci_write_config8(MEMCTRL, (0x48 + i), dl);
dl = dh;
}
}
dh <<= 2;
// F7_Rx57 Ending address mirror register
pci_write_config8(PCI_DEV(0, 0, 7), 0x57, dh);
dev = pci_locate_device(PCI_ID(0x1106, 0x324e), 0);
pci_write_config8(dev, 0x57, dh);
// LOW TOP Address
pci_write_config8(MEMCTRL, 0x88, dh);
pci_write_config8(MEMCTRL, 0x85, dh);
// also program vlink mirror
pci_write_config8(PCI_DEV(0, 0, 7), 0xe5, dh);
/****************************************************************/
/* Set Physical to Virtual Rank mapping */
/****************************************************************/
pci_write_config32(MEMCTRL, 0x54, 0x0);
for (i = 0; i < 4; i++) {
reg8 = pci_read_config8(PCI_DEV(0, 0, 4), (SCRATCH_RANK_0 + i));
if (reg8) {
reg8 = pci_read_config8(MEMCTRL, (0x54 + (i >> 1)));
if (i & 0x1) { /* Odd Rank */
reg8 &= 0xf0;
reg8 |= (0x8 | i);
} else { /* Even Rank */
reg8 &= 0x0f;
reg8 |= ((0x8 | i) << 4);
}
pci_write_config8(MEMCTRL, (0x54 + (i >> 1)), reg8);
}
}
/****************************************************************/
/* Set DRAM Refresh Counter */
/****************************************************************/
val = pci_read_config8(MEMCTRL, 0X90) & 0X7;
val <<= 1;
reg8 = pci_read_config8(PCI_DEV(0, 0, 4), SCRATCH_DRAM_256M_BIT);
if (reg8)
val++;
pci_write_config8(MEMCTRL, 0x6a, REFC_Table[val]);
/****************************************************************/
/* Chipset Performance UP and other setting after DRAM Sizing */
/****************************************************************/
/* Dram Registers */
val = sizeof(Dram_Table) / sizeof(Dram_Table[0]);
for (i = 0; i < val; i += 3) {
reg8 = pci_read_config8(MEMCTRL, Dram_Table[i]);
reg8 &= Dram_Table[i + 1];
reg8 |= Dram_Table[i + 2];
pci_write_config8(MEMCTRL, Dram_Table[i], reg8);
}
/* Host Registers */
val = sizeof(Host_Table) / sizeof(Host_Table[0]);
for (i = 0; i < val; i += 3) {
reg8 = pci_read_config8(HOSTCTRL, Host_Table[i]);
reg8 &= Host_Table[i + 1];
reg8 |= Host_Table[i + 2];
pci_write_config8(HOSTCTRL, Host_Table[i], reg8);
}
/* PM Registers */
#ifdef SETUP_PM_REGISTERS
val = sizeof(PM_Table) / sizeof(PM_Table[0]);
for (i = 0; i < val; i += 3) {
reg8 = pci_read_config8(PCI_DEV(0, 0, 4), PM_Table[i]);
reg8 &= PM_Table[i + 1];
reg8 |= PM_Table[i + 2];
pci_write_config8(PCI_DEV(0, 0, 4), PM_Table[i], reg8);
}
#endif
pci_write_config8(HOSTCTRL, 0x5d, 0xb2);
/****************************************************************/
/* UMA registers for N-series projects */
/****************************************************************/
/* Manual setting frame buffer bank */
for (i = 0; i < 4; i++) {
reg8 = pci_read_config8(PCI_DEV(0, 0, 4), (SCRATCH_RANK_0 + i));
if (reg8)
val = i;
}
pci_write_config8(MEMCTRL, 0xb0, val);
reg8 = 0x40; // Frame buffer size 64M
reg8 |= 0x80; // VGA Enable
reg8 |= 0x0a; // A[31:28] = 1010b
pci_write_config8(MEMCTRL, 0xa1, reg8);
#ifdef ECC
// Clear Ecc
outl(0x80000180, 0xcf8);
outb(0xff, 0xcfc);
// Enable Ecc
outl(0x80000188, 0xcf8);
outb(0xcf, 0xcfc);
reg8 = pci_read_config8(PCI_DEV(0, 0, 0), 0xa5);
reg8 |= 0x10;
pci_write_config8(PCI_DEV(0, 0, 0), 0xa5, reg8);
reg8 = pci_read_config8(PCI_DEV(0, 0, 0), 0x91);
reg8 |= 0x20;
pci_write_config8(PCI_DEV(0, 0, 0), 0x91, reg8);
#endif
static const struct regmask {
u8 reg;
u8 mask;
u8 val;
} b0d1f0[] = {
{ 0x40, 0x00, 0x8b},
{ 0x41, 0x80, 0x43},
{ 0x42, 0x00, 0x62},
{ 0x43, 0x00, 0x44},
{ 0x44, 0x00, 0x34},
{ 0x45, 0x00, 0x72}
}, b0d0f3[] = {
{ 0x53, 0xf0, 0x0f},
{ 0x60, 0x00, 0x03},
{ 0x65, 0x00, 0xd9},
{ 0x66, 0x00, 0x80},
{ 0x67, 0x00, 0x00},
{ 0x68, 0x00, 0x01},
{ 0x69, 0xe0, 0x03},
{ 0x6b, 0x00, 0x10},
{ 0x6c, 0xc1, 0x08},
{ 0x6e, 0x00, 0x89},
{ 0x6f, 0x00, 0x51},
{ 0x75, ~0x40, 0x40},
{ 0x76, 0x8f, 0x00},
{ 0x7b, 0x00, 0xa0},
{ 0x86, 0x01, 0x24},
{ 0x86, 0x04, 0x29},
{ 0x8c, 0x00, 0x00},
{ 0x8d, 0x00, 0x00},
{ 0x95, ~0x40, 0x00},
{ 0xa2, 0x00, 0x44},
{ 0xb1, 0x00, 0xaa}
}, b0d0f0[] = {
{ 0x4d, 0x00, 0x24},
{ 0x4f, 0x00, 0x01},
{ 0xbc, 0x00, 0x21},
{ 0xbe, 0x00, 0x00},
{ 0xbf, 0x7f, 0x80}
}, b0d17f0[] = {
{ 0x40, ~0x01, 0x01}, // enable timer/counter shadow registers
{ 0x67, ~0x03, 0x01},
{ 0x5b, ~0x01, 0x00},
{ 0x8d, ~0x02, 0x02},
{ 0x97, 0x7f, 0x00},
{ 0xd2, ~0x18, 0x00},
{ 0xe2, ~0x36, 0x06},
{ 0xe4, 0x7f, 0x00},
{ 0xe5, 0x00, 0x40},
{ 0xe6, 0x00, 0x20},
{ 0xe7, 0x2f, 0xc0},
{ 0xec, ~0x08, 0x00}
}, b0d17f7[] = {
{ 0x4e, 0x7f, 0x80},
{ 0x4f, ~(1 << 6), 1 << 6 }, /* PG_CX700: 14.1.1 enable P2P Bridge Header for External PCI Bus */
{ 0x74, ~0x00, 0x04}, /* PG_CX700: 14.1.2 APIC FSB directly up to snmic, not on pci */
{ 0x7c, ~0x00, 0x02}, /* PG_CX700: 14.1.1 APIC FSB directly up to snmic, not on pci */
{ 0xe6, 0x0, 0x04} // MSI post
}, b0d19f0[] = { /* P2PE */
{ 0x42, ~0x08, 0x08}, // Disable HD Audio,
{ 0x40, 0x3f, 0x80} // 14.1.3.1.1 of the PG: extended cfg mode for pcie. enable capability, but don't activate
}, b0d0f2[] = {
{ 0x50, ~0x40, 0x88},
{ 0x51, 0x80, 0x7b},
{ 0x52, 0x90, 0x6f},
{ 0x53, 0x00, 0x88},
{ 0x54, 0xe4, 0x16},
{ 0x55, 0xf2, 0x04},
{ 0x56, 0x0f, 0x00},
{ 0x57, ~0x04, 0x00},
{ 0x5d, 0x00, 0xb2},
{ 0x5e, 0x00, 0x88},
{ 0x5f, 0x00, 0xc7},
{ 0x5c, 0x00, 0x01}
};
REGISTERPRESET(0, 0, 0, b0d0f0);
REGISTERPRESET(0, 0, 2, b0d0f2);
REGISTERPRESET(0, 0, 3, b0d0f3);
REGISTERPRESET(0, 1, 0, b0d1f0);
REGISTERPRESET(0, 17, 0, b0d17f0);
REGISTERPRESET(0, 17, 7, b0d17f7);
REGISTERPRESET(0, 19, 0, b0d19f0);
}