nb/intel/haswell: Consolidate memory-down SPD handling

Mainboards do not need to know about `pei_data` to tell northbridge code
where to find the SPD data. Adjust `mb_get_spd_map` to take a pointer to
a struct instead of an array, and update all the mainboards accordingly.

Currently, the only board with memory-down in the tree is google/slippy.
Mainboard code now obtains the SPD index in `mb_get_spd_map` and adjusts
the channel population accordingly. Then, northbridge code reads the SPD
file and uses the index that was read in `mb_get_spd_map`, and copies it
to channel 0 slot 0 unconditionally. MRC only uses the first position of
the `spd_data` array, and ignores the other positions. In coreboot code,
`setup_sdram_meminfo` uses the data of each SPD index, so `copy_spd` has
to account for this.

Tested on Asrock B85M Pro4, still boots and still resumes from S3.

Change-Id: Ibaed5c6de9853db6abd08f53bbfda8800d207c3e
Signed-off-by: Angel Pons <th3fanbus@gmail.com>
Reviewed-on: https://review.coreboot.org/c/coreboot/+/51448
Reviewed-by: Nico Huber <nico.h@gmx.de>
Reviewed-by: Matt DeVillier <matt.devillier@gmail.com>
Tested-by: build bot (Jenkins) <no-reply@coreboot.org>
This commit is contained in:
Angel Pons 2021-03-12 17:00:52 +01:00 committed by Patrick Georgi
parent afc6c0ae12
commit 90ae08922d
16 changed files with 114 additions and 114 deletions

View file

@ -17,12 +17,12 @@ void mainboard_config_rcba(void)
RCBA16(D20IR) = DIR_ROUTE(PIRQA, PIRQB, PIRQC, PIRQD); RCBA16(D20IR) = DIR_ROUTE(PIRQA, PIRQB, PIRQC, PIRQD);
} }
void mb_get_spd_map(uint8_t spd_map[4]) void mb_get_spd_map(struct spd_info *spdi)
{ {
spd_map[0] = 0xa0; spdi->addresses[0] = 0xa0;
spd_map[1] = 0xa2; spdi->addresses[1] = 0xa2;
spd_map[2] = 0xa4; spdi->addresses[2] = 0xa4;
spd_map[3] = 0xa6; spdi->addresses[3] = 0xa6;
} }
const struct usb2_port_setting mainboard_usb2_ports[MAX_USB2_PORTS] = { const struct usb2_port_setting mainboard_usb2_ports[MAX_USB2_PORTS] = {

View file

@ -17,10 +17,10 @@ void mainboard_config_rcba(void)
RCBA16(D20IR) = DIR_ROUTE(PIRQA, PIRQB, PIRQC, PIRQD); RCBA16(D20IR) = DIR_ROUTE(PIRQA, PIRQB, PIRQC, PIRQD);
} }
void mb_get_spd_map(uint8_t spd_map[4]) void mb_get_spd_map(struct spd_info *spdi)
{ {
spd_map[0] = 0xa0; spdi->addresses[0] = 0xa0;
spd_map[2] = 0xa4; spdi->addresses[2] = 0xa4;
} }
const struct usb2_port_setting mainboard_usb2_ports[MAX_USB2_PORTS] = { const struct usb2_port_setting mainboard_usb2_ports[MAX_USB2_PORTS] = {

View file

@ -40,10 +40,10 @@ void mainboard_config_rcba(void)
RCBA16(D23IR) = DIR_ROUTE(PIRQH, PIRQH, PIRQH, PIRQH); /* SDIO */ RCBA16(D23IR) = DIR_ROUTE(PIRQH, PIRQH, PIRQH, PIRQH); /* SDIO */
} }
void mb_get_spd_map(uint8_t spd_map[4]) void mb_get_spd_map(struct spd_info *spdi)
{ {
spd_map[0] = 0xa0; spdi->addresses[0] = 0xa0;
spd_map[2] = 0xa4; spdi->addresses[2] = 0xa4;
} }
const struct usb2_port_setting mainboard_usb2_ports[MAX_USB2_PORTS] = { const struct usb2_port_setting mainboard_usb2_ports[MAX_USB2_PORTS] = {

View file

@ -1,12 +1,7 @@
/* SPDX-License-Identifier: GPL-2.0-only */ /* SPDX-License-Identifier: GPL-2.0-only */
#include <cbfs.h>
#include <console/console.h>
#include <northbridge/intel/haswell/haswell.h>
#include <northbridge/intel/haswell/raminit.h> #include <northbridge/intel/haswell/raminit.h>
#include <southbridge/intel/lynxpoint/pch.h> #include <southbridge/intel/lynxpoint/pch.h>
#include <string.h>
#include <types.h>
#include "variant.h" #include "variant.h"
@ -45,31 +40,9 @@ void mainboard_config_rcba(void)
RCBA16(D23IR) = DIR_ROUTE(PIRQH, PIRQH, PIRQH, PIRQH); /* SDIO */ RCBA16(D23IR) = DIR_ROUTE(PIRQH, PIRQH, PIRQH, PIRQH); /* SDIO */
} }
void mb_get_spd_map(uint8_t spd_map[4]) void mb_get_spd_map(struct spd_info *spdi)
{ {
spd_map[0] = 0xff; spdi->spd_index = variant_get_spd_index();
spd_map[2] = 0xff; spdi->addresses[0] = SPD_MEMORY_DOWN;
} spdi->addresses[2] = variant_is_dual_channel(spdi->spd_index) ? SPD_MEMORY_DOWN : 0;
unsigned int fill_spd_for_index(uint8_t spd[], unsigned int spd_index)
{
uint8_t *spd_file;
size_t spd_file_len;
printk(BIOS_DEBUG, "SPD index %d\n", spd_index);
spd_file = cbfs_map("spd.bin", &spd_file_len);
if (!spd_file)
die("SPD data not found.");
if (spd_file_len < ((spd_index + 1) * SPD_LEN)) {
printk(BIOS_ERR, "SPD index override to 0 - old hardware?\n");
spd_index = 0;
}
if (spd_file_len < SPD_LEN)
die("Missing SPD data.");
memcpy(spd, spd_file + (spd_index * SPD_LEN), SPD_LEN);
return spd_index;
} }

View file

@ -3,8 +3,9 @@
#ifndef VARIANT_H #ifndef VARIANT_H
#define VARIANT_H #define VARIANT_H
#include <stdint.h> #include <types.h>
unsigned int fill_spd_for_index(uint8_t spd[], unsigned int index); unsigned int variant_get_spd_index(void);
bool variant_is_dual_channel(const unsigned int spd_index);
#endif #endif

View file

@ -6,22 +6,22 @@
#include <southbridge/intel/lynxpoint/lp_gpio.h> #include <southbridge/intel/lynxpoint/lp_gpio.h>
#include "../../variant.h" #include "../../variant.h"
/* Copy SPD data for on-board memory */ unsigned int variant_get_spd_index(void)
void copy_spd(struct pei_data *peid)
{ {
const int gpio_vector[] = {13, 9, 47, -1}; const int gpio_vector[] = {13, 9, 47, -1};
return get_gpios(gpio_vector);
}
unsigned int spd_index = fill_spd_for_index(peid->spd_data[0], get_gpios(gpio_vector)); bool variant_is_dual_channel(const unsigned int spd_index)
{
/* Index 0-2,6 are 4GB config with both CH0 and CH1 /* Index 0-2,6 are 4GB config with both CH0 and CH1
* Index 3-5,7 are 2GB config with CH0 only Index 3-5,7 are 2GB config with CH0 only */
*/
switch (spd_index) { switch (spd_index) {
case 0: case 1: case 2: case 6: case 0: case 1: case 2: case 6:
memcpy(peid->spd_data[2], peid->spd_data[0], SPD_LEN); return true;
break;
case 3: case 4: case 5: case 7: case 3: case 4: case 5: case 7:
peid->dimm_channel1_disabled = 3; default:
return false;
} }
} }

View file

@ -6,20 +6,17 @@
#include <southbridge/intel/lynxpoint/lp_gpio.h> #include <southbridge/intel/lynxpoint/lp_gpio.h>
#include "../../variant.h" #include "../../variant.h"
/* Copy SPD data for on-board memory */ unsigned int variant_get_spd_index(void)
void copy_spd(struct pei_data *peid)
{ {
const int gpio_vector[] = {13, 9, 47, -1}; const int gpio_vector[] = {13, 9, 47, -1};
return get_gpios(gpio_vector);
}
unsigned int spd_index = fill_spd_for_index(peid->spd_data[0], get_gpios(gpio_vector)); bool variant_is_dual_channel(const unsigned int spd_index)
{
/* Limiting to a single dimm for 2GB configuration /* Limiting to a single dimm for 2GB configuration
* Identified by bit 3 Identified by bit 2 */
*/ return !(spd_index & 0x4);
if (spd_index & 0x4)
peid->dimm_channel1_disabled = 3;
else
memcpy(peid->spd_data[2], peid->spd_data[0], SPD_LEN);
} }
const struct usb2_port_setting mainboard_usb2_ports[MAX_USB2_PORTS] = { const struct usb2_port_setting mainboard_usb2_ports[MAX_USB2_PORTS] = {

View file

@ -9,33 +9,26 @@
#include "../../onboard.h" #include "../../onboard.h"
#include "../../variant.h" #include "../../variant.h"
/* Copy SPD data for on-board memory */ unsigned int variant_get_spd_index(void)
void copy_spd(struct pei_data *peid)
{ {
const int gpio_vector[] = {13, 9, 47, -1}; const int gpio_vector[] = {13, 9, 47, -1};
return get_gpios(gpio_vector);
}
unsigned int spd_index = fill_spd_for_index(peid->spd_data[0], get_gpios(gpio_vector)); bool variant_is_dual_channel(const unsigned int spd_index)
{
uint32_t board_version = PEPPY_BOARD_VERSION_PROTO; uint32_t board_version = PEPPY_BOARD_VERSION_PROTO;
google_chromeec_get_board_version(&board_version); google_chromeec_get_board_version(&board_version);
switch (board_version) { switch (board_version) {
case PEPPY_BOARD_VERSION_PROTO: case PEPPY_BOARD_VERSION_PROTO:
/* Index 0 is 2GB config with CH0 only. */ /* Index 0 is 2GB config with CH0 only. */
if (spd_index == 0) return spd_index != 0;
peid->dimm_channel1_disabled = 3;
else
memcpy(peid->spd_data[2], peid->spd_data[0], SPD_LEN);
break;
case PEPPY_BOARD_VERSION_EVT: case PEPPY_BOARD_VERSION_EVT:
default: default:
/* Index 0-3 are 4GB config with both CH0 and CH1. /* Index 0-3 are 4GB config with both CH0 and CH1.
* Index 4-7 are 2GB config with CH0 only. */ Index 4-7 are 2GB config with CH0 only. */
if (spd_index > 3) return spd_index <= 3;
peid->dimm_channel1_disabled = 3;
else
memcpy(peid->spd_data[2], peid->spd_data[0], SPD_LEN);
break;
} }
} }

View file

@ -6,22 +6,22 @@
#include <southbridge/intel/lynxpoint/lp_gpio.h> #include <southbridge/intel/lynxpoint/lp_gpio.h>
#include "../../variant.h" #include "../../variant.h"
/* Copy SPD data for on-board memory */ unsigned int variant_get_spd_index(void)
void copy_spd(struct pei_data *peid)
{ {
const int gpio_vector[] = {13, 9, 47, -1}; const int gpio_vector[] = {13, 9, 47, -1};
return get_gpios(gpio_vector);
}
unsigned int spd_index = fill_spd_for_index(peid->spd_data[0], get_gpios(gpio_vector)); bool variant_is_dual_channel(const unsigned int spd_index)
{
/* Index 0-2, are 4GB config with both CH0 and CH1 /* Index 0-2 are 4GB config with both CH0 and CH1
* Index 3-5, are 2GB config with CH0 only Index 3-5 are 2GB config with CH0 only */
*/
switch (spd_index) { switch (spd_index) {
case 0: case 1: case 2: case 0: case 1: case 2:
memcpy(peid->spd_data[2], peid->spd_data[0], SPD_LEN); return true;
break;
case 3: case 4: case 5: case 3: case 4: case 5:
peid->dimm_channel1_disabled = 3; default:
return false;
} }
} }

View file

@ -17,10 +17,10 @@ void mainboard_config_rcba(void)
RCBA16(D20IR) = DIR_ROUTE(PIRQA, PIRQB, PIRQC, PIRQD); RCBA16(D20IR) = DIR_ROUTE(PIRQA, PIRQB, PIRQC, PIRQD);
} }
void mb_get_spd_map(uint8_t spd_map[4]) void mb_get_spd_map(struct spd_info *spdi)
{ {
spd_map[0] = 0xa0; spdi->addresses[0] = 0xa0;
spd_map[2] = 0xa4; spdi->addresses[2] = 0xa4;
} }
const struct usb2_port_setting mainboard_usb2_ports[MAX_USB2_PORTS] = { const struct usb2_port_setting mainboard_usb2_ports[MAX_USB2_PORTS] = {

View file

@ -41,12 +41,12 @@ void mainboard_config_rcba(void)
RCBA16(D22IR) = DIR_ROUTE(PIRQA, PIRQB, PIRQC, PIRQD); RCBA16(D22IR) = DIR_ROUTE(PIRQA, PIRQB, PIRQC, PIRQD);
} }
void mb_get_spd_map(uint8_t spd_map[4]) void mb_get_spd_map(struct spd_info *spdi)
{ {
spd_map[0] = 0xa0; spdi->addresses[0] = 0xa0;
spd_map[1] = 0xa2; spdi->addresses[1] = 0xa2;
spd_map[2] = 0xa4; spdi->addresses[2] = 0xa4;
spd_map[3] = 0xa6; spdi->addresses[3] = 0xa6;
} }
const struct usb2_port_setting mainboard_usb2_ports[MAX_USB2_PORTS] = { const struct usb2_port_setting mainboard_usb2_ports[MAX_USB2_PORTS] = {

View file

@ -40,10 +40,10 @@ void mb_late_romstage_setup(void)
} }
} }
void mb_get_spd_map(uint8_t spd_map[4]) void mb_get_spd_map(struct spd_info *spdi)
{ {
spd_map[0] = 0xa0; spdi->addresses[0] = 0xa0;
spd_map[2] = 0xa2; spdi->addresses[2] = 0xa2;
} }
const struct usb2_port_setting mainboard_usb2_ports[MAX_USB2_PORTS] = { const struct usb2_port_setting mainboard_usb2_ports[MAX_USB2_PORTS] = {

View file

@ -17,10 +17,10 @@ void mainboard_config_rcba(void)
RCBA16(D20IR) = DIR_ROUTE(PIRQA, PIRQB, PIRQC, PIRQD); RCBA16(D20IR) = DIR_ROUTE(PIRQA, PIRQB, PIRQC, PIRQD);
} }
void mb_get_spd_map(uint8_t spd_map[4]) void mb_get_spd_map(struct spd_info *spdi)
{ {
spd_map[0] = 0xa0; spdi->addresses[0] = 0xa0;
spd_map[2] = 0xa4; spdi->addresses[2] = 0xa4;
} }
const struct usb2_port_setting mainboard_usb2_ports[MAX_USB2_PORTS] = { const struct usb2_port_setting mainboard_usb2_ports[MAX_USB2_PORTS] = {

View file

@ -17,12 +17,12 @@ void mainboard_config_rcba(void)
RCBA16(D20IR) = DIR_ROUTE(PIRQA, PIRQB, PIRQC, PIRQD); RCBA16(D20IR) = DIR_ROUTE(PIRQA, PIRQB, PIRQC, PIRQD);
} }
void mb_get_spd_map(uint8_t spd_map[4]) void mb_get_spd_map(struct spd_info *spdi)
{ {
spd_map[0] = 0xa0; spdi->addresses[0] = 0xa0;
spd_map[1] = 0xa2; spdi->addresses[1] = 0xa2;
spd_map[2] = 0xa4; spdi->addresses[2] = 0xa4;
spd_map[3] = 0xa6; spdi->addresses[3] = 0xa6;
} }
const struct usb2_port_setting mainboard_usb2_ports[MAX_USB2_PORTS] = { const struct usb2_port_setting mainboard_usb2_ports[MAX_USB2_PORTS] = {

View file

@ -6,15 +6,19 @@
#include <stdint.h> #include <stdint.h>
#include "pei_data.h" #include "pei_data.h"
#define SPD_MEMORY_DOWN 0xff
struct spd_info {
uint8_t addresses[4];
unsigned int spd_index;
};
/* Mainboard-specific USB configuration */ /* Mainboard-specific USB configuration */
extern const struct usb2_port_setting mainboard_usb2_ports[MAX_USB2_PORTS]; extern const struct usb2_port_setting mainboard_usb2_ports[MAX_USB2_PORTS];
extern const struct usb3_port_setting mainboard_usb3_ports[MAX_USB3_PORTS]; extern const struct usb3_port_setting mainboard_usb3_ports[MAX_USB3_PORTS];
/* Optional function to copy SPD data for on-board memory */
void copy_spd(struct pei_data *peid);
/* Mainboard callback to fill in the SPD addresses in MRC format */ /* Mainboard callback to fill in the SPD addresses in MRC format */
void mb_get_spd_map(uint8_t spd_map[4]); void mb_get_spd_map(struct spd_info *spdi);
void sdram_initialize(struct pei_data *pei_data); void sdram_initialize(struct pei_data *pei_data);
void setup_sdram_meminfo(struct pei_data *pei_data); void setup_sdram_meminfo(struct pei_data *pei_data);

View file

@ -1,6 +1,7 @@
/* SPDX-License-Identifier: GPL-2.0-only */ /* SPDX-License-Identifier: GPL-2.0-only */
#include <arch/romstage.h> #include <arch/romstage.h>
#include <cbfs.h>
#include <console/console.h> #include <console/console.h>
#include <cf9_reset.h> #include <cf9_reset.h>
#include <device/device.h> #include <device/device.h>
@ -21,10 +22,37 @@
#include <southbridge/intel/lynxpoint/pch.h> #include <southbridge/intel/lynxpoint/pch.h>
#include <southbridge/intel/lynxpoint/me.h> #include <southbridge/intel/lynxpoint/me.h>
#include <string.h> #include <string.h>
#include <types.h>
/* Copy SPD data for on-board memory */ /* Copy SPD data for on-board memory */
void __weak copy_spd(struct pei_data *peid) static void copy_spd(struct pei_data *pei_data, struct spd_info *spdi)
{ {
if (!CONFIG(HAVE_SPD_IN_CBFS))
return;
printk(BIOS_DEBUG, "SPD index %d\n", spdi->spd_index);
size_t spd_file_len;
uint8_t *spd_file = cbfs_map("spd.bin", &spd_file_len);
if (!spd_file)
die("SPD data not found.");
if (spd_file_len < ((spdi->spd_index + 1) * SPD_LEN)) {
printk(BIOS_ERR, "SPD index override to 0 - old hardware?\n");
spdi->spd_index = 0;
}
if (spd_file_len < SPD_LEN)
die("Missing SPD data.");
/* MRC only uses index 0, but coreboot uses the other indices */
memcpy(pei_data->spd_data[0], spd_file + (spdi->spd_index * SPD_LEN), SPD_LEN);
for (size_t i = 1; i < ARRAY_SIZE(spdi->addresses); i++) {
if (spdi->addresses[i] == SPD_MEMORY_DOWN)
memcpy(pei_data->spd_data[i], pei_data->spd_data[0], SPD_LEN);
}
} }
void __weak mb_late_romstage_setup(void) void __weak mb_late_romstage_setup(void)
@ -98,7 +126,11 @@ void mainboard_romstage_entry(void)
pei_data.boot_mode = s3resume ? 2 : 0; pei_data.boot_mode = s3resume ? 2 : 0;
/* Obtain the SPD addresses from mainboard code */ /* Obtain the SPD addresses from mainboard code */
mb_get_spd_map(pei_data.spd_addresses); struct spd_info spdi = {0};
mb_get_spd_map(&spdi);
for (size_t i = 0; i < ARRAY_SIZE(spdi.addresses); i++)
pei_data.spd_addresses[i] = spdi.addresses[i];
/* Calculate unimplemented DIMM slots for each channel */ /* Calculate unimplemented DIMM slots for each channel */
pei_data.dimm_channel0_disabled = make_channel_disabled_mask(&pei_data, 0); pei_data.dimm_channel0_disabled = make_channel_disabled_mask(&pei_data, 0);
@ -111,7 +143,7 @@ void mainboard_romstage_entry(void)
if (CONFIG(INTEL_TXT)) if (CONFIG(INTEL_TXT))
intel_txt_romstage_init(); intel_txt_romstage_init();
copy_spd(&pei_data); copy_spd(&pei_data, &spdi);
sdram_initialize(&pei_data); sdram_initialize(&pei_data);