1
0
mirror of https://review.coreboot.org/flashrom.git synced 2025-10-22 10:10:52 +02:00

ichspi: Add Apollo Lake support

It's almost identical to 100 series PCHs and later. There are some
additional FREGs (12..15). To not clutter the `if` conditions further,
make more use of `switch` statements.

Tested on Kontron mAL10. Mark it as DEP as usually the last sector
is not covered by the descriptor layout and can't be read.

Change-Id: I1c464b5b3d151e6d28d5db96495fe874a0a45718
Signed-off-by: Nico Huber <nico.huber@secunet.com>
Reviewed-on: https://review.coreboot.org/c/flashrom/+/30995
Tested-by: build bot (Jenkins) <no-reply@coreboot.org>
Reviewed-by: Angel Pons <th3fanbus@gmail.com>
This commit is contained in:
Nico Huber
2019-01-18 16:49:37 +01:00
committed by Nico Huber
parent 3750986348
commit d2d3993a25
4 changed files with 157 additions and 63 deletions

View File

@@ -2026,7 +2026,7 @@ const struct penable chipset_enables[] = {
{0x8086, 0xa2c8, B_S, NT, "Intel", "B250", enable_flash_pch100}, {0x8086, 0xa2c8, B_S, NT, "Intel", "B250", enable_flash_pch100},
{0x8086, 0xa2c9, B_S, NT, "Intel", "Z370", enable_flash_pch100}, {0x8086, 0xa2c9, B_S, NT, "Intel", "Z370", enable_flash_pch100},
{0x8086, 0xa2d2, B_S, NT, "Intel", "X299", enable_flash_pch100}, {0x8086, 0xa2d2, B_S, NT, "Intel", "X299", enable_flash_pch100},
{0x8086, 0x5ae8, B_S, BAD, "Intel", "Apollo Lake", enable_flash_apl}, {0x8086, 0x5ae8, B_S, DEP, "Intel", "Apollo Lake", enable_flash_apl},
#endif #endif
{0}, {0},
}; };

View File

@@ -42,6 +42,8 @@
ssize_t ich_number_of_regions(const enum ich_chipset cs, const struct ich_desc_content *const cont) ssize_t ich_number_of_regions(const enum ich_chipset cs, const struct ich_desc_content *const cont)
{ {
switch (cs) { switch (cs) {
case CHIPSET_APOLLO_LAKE:
return 6;
case CHIPSET_C620_SERIES_LEWISBURG: case CHIPSET_C620_SERIES_LEWISBURG:
return 16; return 16;
case CHIPSET_100_SERIES_SUNRISE_POINT: case CHIPSET_100_SERIES_SUNRISE_POINT:
@@ -67,6 +69,7 @@ ssize_t ich_number_of_masters(const enum ich_chipset cs, const struct ich_desc_c
{ {
switch (cs) { switch (cs) {
case CHIPSET_C620_SERIES_LEWISBURG: case CHIPSET_C620_SERIES_LEWISBURG:
case CHIPSET_APOLLO_LAKE:
if (cont->NM <= MAX_NUM_MASTERS) if (cont->NM <= MAX_NUM_MASTERS)
return cont->NM; return cont->NM;
break; break;
@@ -103,7 +106,7 @@ void prettyprint_ich_chipset(enum ich_chipset cs)
"5 series Ibex Peak", "6 series Cougar Point", "7 series Panther Point", "5 series Ibex Peak", "6 series Cougar Point", "7 series Panther Point",
"8 series Lynx Point", "Baytrail", "8 series Lynx Point LP", "8 series Wellsburg", "8 series Lynx Point", "Baytrail", "8 series Lynx Point LP", "8 series Wellsburg",
"9 series Wildcat Point", "9 series Wildcat Point LP", "100 series Sunrise Point", "9 series Wildcat Point", "9 series Wildcat Point LP", "100 series Sunrise Point",
"C620 series Lewisburg", "C620 series Lewisburg", "Apollo Lake",
}; };
if (cs < CHIPSET_ICH8 || cs - CHIPSET_ICH8 + 1 >= ARRAY_SIZE(chipset_names)) if (cs < CHIPSET_ICH8 || cs - CHIPSET_ICH8 + 1 >= ARRAY_SIZE(chipset_names))
cs = 0; cs = 0;
@@ -140,8 +143,8 @@ void prettyprint_ich_descriptor_content(enum ich_chipset cs, const struct ich_de
msg_pdbg2("FRBA (Flash Region Base Address): 0x%03x\n", getFRBA(cont)); msg_pdbg2("FRBA (Flash Region Base Address): 0x%03x\n", getFRBA(cont));
msg_pdbg2("NC (Number of Components): %5d\n", cont->NC + 1); msg_pdbg2("NC (Number of Components): %5d\n", cont->NC + 1);
msg_pdbg2("FCBA (Flash Component Base Address): 0x%03x\n", getFCBA(cont)); msg_pdbg2("FCBA (Flash Component Base Address): 0x%03x\n", getFCBA(cont));
msg_pdbg2("ISL (ICH/PCH Strap Length): %5d\n", cont->ISL); msg_pdbg2("ISL (ICH/PCH/SoC Strap Length): %5d\n", cont->ISL);
msg_pdbg2("FISBA/FPSBA (Flash ICH/PCH Strap Base Address): 0x%03x\n", getFISBA(cont)); msg_pdbg2("FISBA/FPSBA (Flash ICH/PCH/SoC Strap Base Addr): 0x%03x\n", getFISBA(cont));
msg_pdbg2("NM (Number of Masters): %5zd\n", ich_number_of_masters(cs, cont)); msg_pdbg2("NM (Number of Masters): %5zd\n", ich_number_of_masters(cs, cont));
msg_pdbg2("FMBA (Flash Master Base Address): 0x%03x\n", getFMBA(cont)); msg_pdbg2("FMBA (Flash Master Base Address): 0x%03x\n", getFMBA(cont));
msg_pdbg2("MSL/PSL (MCH/PROC Strap Length): %5d\n", cont->MSL); msg_pdbg2("MSL/PSL (MCH/PROC Strap Length): %5d\n", cont->MSL);
@@ -194,7 +197,8 @@ static const char *pprint_density(enum ich_chipset cs, const struct ich_descript
case CHIPSET_9_SERIES_WILDCAT_POINT: case CHIPSET_9_SERIES_WILDCAT_POINT:
case CHIPSET_9_SERIES_WILDCAT_POINT_LP: case CHIPSET_9_SERIES_WILDCAT_POINT_LP:
case CHIPSET_100_SERIES_SUNRISE_POINT: case CHIPSET_100_SERIES_SUNRISE_POINT:
case CHIPSET_C620_SERIES_LEWISBURG: { case CHIPSET_C620_SERIES_LEWISBURG:
case CHIPSET_APOLLO_LAKE: {
uint8_t size_enc; uint8_t size_enc;
if (idx == 0) { if (idx == 0) {
size_enc = desc->component.dens_new.comp1_density; size_enc = desc->component.dens_new.comp1_density;
@@ -213,7 +217,7 @@ static const char *pprint_density(enum ich_chipset cs, const struct ich_descript
static const char *pprint_freq(enum ich_chipset cs, uint8_t value) static const char *pprint_freq(enum ich_chipset cs, uint8_t value)
{ {
static const char *const freq_str[2][8] = { { static const char *const freq_str[3][8] = { {
"20 MHz", "20 MHz",
"33 MHz", "33 MHz",
"reserved", "reserved",
@@ -231,6 +235,15 @@ static const char *pprint_freq(enum ich_chipset cs, uint8_t value)
"reserved", "reserved",
"17 MHz", "17 MHz",
"reserved" "reserved"
}, {
"reserved",
"50 MHz",
"40 MHz",
"reserved",
"25 MHz",
"reserved",
"14 MHz / 17 MHz",
"reserved"
} }; } };
switch (cs) { switch (cs) {
@@ -253,6 +266,8 @@ static const char *pprint_freq(enum ich_chipset cs, uint8_t value)
case CHIPSET_100_SERIES_SUNRISE_POINT: case CHIPSET_100_SERIES_SUNRISE_POINT:
case CHIPSET_C620_SERIES_LEWISBURG: case CHIPSET_C620_SERIES_LEWISBURG:
return freq_str[1][value]; return freq_str[1][value];
case CHIPSET_APOLLO_LAKE:
return freq_str[2][value];
case CHIPSET_ICH_UNKNOWN: case CHIPSET_ICH_UNKNOWN:
default: default:
return "unknown"; return "unknown";
@@ -261,11 +276,23 @@ static const char *pprint_freq(enum ich_chipset cs, uint8_t value)
void prettyprint_ich_descriptor_component(enum ich_chipset cs, const struct ich_descriptors *desc) void prettyprint_ich_descriptor_component(enum ich_chipset cs, const struct ich_descriptors *desc)
{ {
bool has_flill1;
switch (cs) {
case CHIPSET_100_SERIES_SUNRISE_POINT:
case CHIPSET_C620_SERIES_LEWISBURG:
case CHIPSET_APOLLO_LAKE:
has_flill1 = true;
break;
default:
has_flill1 = false;
break;
}
msg_pdbg2("=== Component Section ===\n"); msg_pdbg2("=== Component Section ===\n");
msg_pdbg2("FLCOMP 0x%08x\n", desc->component.FLCOMP); msg_pdbg2("FLCOMP 0x%08x\n", desc->component.FLCOMP);
msg_pdbg2("FLILL 0x%08x\n", desc->component.FLILL ); msg_pdbg2("FLILL 0x%08x\n", desc->component.FLILL );
if (cs == CHIPSET_100_SERIES_SUNRISE_POINT || cs == CHIPSET_C620_SERIES_LEWISBURG) if (has_flill1)
msg_pdbg2("FLILL1 0x%08x\n", desc->component.FLILL1); msg_pdbg2("FLILL1 0x%08x\n", desc->component.FLILL1);
msg_pdbg2("\n"); msg_pdbg2("\n");
@@ -298,7 +325,7 @@ void prettyprint_ich_descriptor_component(enum ich_chipset cs, const struct ich_
msg_pdbg2("Invalid instruction 3: 0x%02x\n", msg_pdbg2("Invalid instruction 3: 0x%02x\n",
desc->component.invalid_instr3); desc->component.invalid_instr3);
} }
if (cs == CHIPSET_100_SERIES_SUNRISE_POINT || cs == CHIPSET_C620_SERIES_LEWISBURG) { if (has_flill1) {
if (desc->component.FLILL1 != 0) { if (desc->component.FLILL1 != 0) {
has_forbidden_opcode = 1; has_forbidden_opcode = 1;
msg_pdbg2("Invalid instruction 4: 0x%02x\n", msg_pdbg2("Invalid instruction 4: 0x%02x\n",
@@ -320,7 +347,7 @@ void prettyprint_ich_descriptor_component(enum ich_chipset cs, const struct ich_
static void pprint_freg(const struct ich_desc_region *reg, uint32_t i) static void pprint_freg(const struct ich_desc_region *reg, uint32_t i)
{ {
static const char *const region_names[] = { static const char *const region_names[] = {
"Descr.", "BIOS", "ME", "GbE", "Platf.", "unknown", "BIOS2", "unknown", "Descr.", "BIOS", "ME", "GbE", "Platf.", "DevExp", "BIOS2", "unknown",
"EC/BMC", "unknown", "IE", "10GbE", "unknown", "unknown", "unknown", "unknown" "EC/BMC", "unknown", "IE", "10GbE", "unknown", "unknown", "unknown", "unknown"
}; };
if (i >= ARRAY_SIZE(region_names)) { if (i >= ARRAY_SIZE(region_names)) {
@@ -417,6 +444,23 @@ void prettyprint_ich_descriptor_master(const enum ich_chipset cs, const struct i
desc->master.mstr[i].write & (1 << j) ? 'w' : ' '); desc->master.mstr[i].write & (1 << j) ? 'w' : ' ');
msg_pdbg2("\n"); msg_pdbg2("\n");
} }
} else if (cs == CHIPSET_APOLLO_LAKE) {
const char *const master_names[] = { "BIOS", "TXE", };
if (nm > ARRAY_SIZE(master_names)) {
msg_pdbg2("%s: number of masters too high (%d).\n", __func__, desc->content.NM);
return;
}
msg_pdbg2(" FD IFWI TXE n/a Platf DevExp\n");
for (i = 0; i < nm; i++) {
size_t j;
msg_pdbg2("%-4s", master_names[i]);
for (j = 0; j < ich_number_of_regions(cs, &desc->content); j++)
msg_pdbg2(" %c%c ",
desc->master.mstr[i].read & (1 << j) ? 'r' : ' ',
desc->master.mstr[i].write & (1 << j) ? 'w' : ' ');
msg_pdbg2("\n");
}
} else { } else {
const struct ich_desc_master *const mstr = &desc->master; const struct ich_desc_master *const mstr = &desc->master;
msg_pdbg2(" Descr. BIOS ME GbE Platf.\n"); msg_pdbg2(" Descr. BIOS ME GbE Platf.\n");
@@ -859,6 +903,11 @@ static enum ich_chipset guess_ich_chipset_from_content(const struct ich_desc_con
return CHIPSET_ICH10; return CHIPSET_ICH10;
else if (content->ISL <= 16) else if (content->ISL <= 16)
return CHIPSET_5_SERIES_IBEX_PEAK; return CHIPSET_5_SERIES_IBEX_PEAK;
else if (content->FLMAP2 == 0) {
if (content->ISL != 19)
msg_pwarn("Peculiar firmware descriptor, assuming Apollo Lake compatibility.\n");
return CHIPSET_APOLLO_LAKE;
}
msg_pwarn("Peculiar firmware descriptor, assuming Ibex Peak compatibility.\n"); msg_pwarn("Peculiar firmware descriptor, assuming Ibex Peak compatibility.\n");
return CHIPSET_5_SERIES_IBEX_PEAK; return CHIPSET_5_SERIES_IBEX_PEAK;
} else if (content->ICCRIBA < 0x31 && content->FMSBA < 0x30) { } else if (content->ICCRIBA < 0x31 && content->FMSBA < 0x30) {
@@ -888,8 +937,20 @@ static enum ich_chipset guess_ich_chipset(const struct ich_desc_content *const c
{ {
const enum ich_chipset guess = guess_ich_chipset_from_content(content); const enum ich_chipset guess = guess_ich_chipset_from_content(content);
if (component->modes.freq_read == 6) { switch (guess) {
if ((guess != CHIPSET_100_SERIES_SUNRISE_POINT) && (guess != CHIPSET_C620_SERIES_LEWISBURG)) { case CHIPSET_100_SERIES_SUNRISE_POINT:
case CHIPSET_C620_SERIES_LEWISBURG:
case CHIPSET_APOLLO_LAKE:
if (component->modes.freq_read != 6) {
msg_pwarn("\nThe firmware descriptor looks like a Skylake/Sunrise Point descriptor.\n"
"However, the read frequency isn't set to 17MHz (the only valid value).\n"
"Please report this message, the output of `ich_descriptors_tool` for\n"
"your descriptor and the output of `lspci -nn` to flashrom@flashrom.org\n\n");
return CHIPSET_9_SERIES_WILDCAT_POINT;
}
return guess;
default:
if (component->modes.freq_read == 6) {
msg_pwarn("\nThe firmware descriptor has the read frequency set to 17MHz. However,\n" msg_pwarn("\nThe firmware descriptor has the read frequency set to 17MHz. However,\n"
"it doesn't look like a Skylake/Sunrise Point compatible descriptor.\n" "it doesn't look like a Skylake/Sunrise Point compatible descriptor.\n"
"Please report this message, the output of `ich_descriptors_tool` for\n" "Please report this message, the output of `ich_descriptors_tool` for\n"
@@ -897,17 +958,7 @@ static enum ich_chipset guess_ich_chipset(const struct ich_desc_content *const c
return CHIPSET_100_SERIES_SUNRISE_POINT; return CHIPSET_100_SERIES_SUNRISE_POINT;
} }
return guess; return guess;
} else {
if (guess == CHIPSET_100_SERIES_SUNRISE_POINT || guess == CHIPSET_C620_SERIES_LEWISBURG) {
msg_pwarn("\nThe firmware descriptor looks like a Skylake/Sunrise Point descriptor.\n"
"However, the read frequency isn't set to 17MHz (the only valid value).\n"
"Please report this message, the output of `ich_descriptors_tool` for\n"
"your descriptor and the output of `lspci -nn` to flashrom@flashrom.org\n\n");
return CHIPSET_9_SERIES_WILDCAT_POINT;
}
} }
return guess;
} }
/* len is the length of dump in bytes */ /* len is the length of dump in bytes */
@@ -1038,6 +1089,7 @@ int getFCBA_component_density(enum ich_chipset cs, const struct ich_descriptors
case CHIPSET_9_SERIES_WILDCAT_POINT_LP: case CHIPSET_9_SERIES_WILDCAT_POINT_LP:
case CHIPSET_100_SERIES_SUNRISE_POINT: case CHIPSET_100_SERIES_SUNRISE_POINT:
case CHIPSET_C620_SERIES_LEWISBURG: case CHIPSET_C620_SERIES_LEWISBURG:
case CHIPSET_APOLLO_LAKE:
if (idx == 0) { if (idx == 0) {
size_enc = desc->component.dens_new.comp1_density; size_enc = desc->component.dens_new.comp1_density;
} else { } else {
@@ -1068,10 +1120,13 @@ static uint32_t read_descriptor_reg(enum ich_chipset cs, uint8_t section, uint16
uint32_t control = 0; uint32_t control = 0;
control |= (section << FDOC_FDSS_OFF) & FDOC_FDSS; control |= (section << FDOC_FDSS_OFF) & FDOC_FDSS;
control |= (offset << FDOC_FDSI_OFF) & FDOC_FDSI; control |= (offset << FDOC_FDSI_OFF) & FDOC_FDSI;
if (cs == CHIPSET_100_SERIES_SUNRISE_POINT || cs == CHIPSET_C620_SERIES_LEWISBURG) { switch (cs) {
case CHIPSET_100_SERIES_SUNRISE_POINT:
case CHIPSET_C620_SERIES_LEWISBURG:
case CHIPSET_APOLLO_LAKE:
mmio_le_writel(control, spibar + PCH100_REG_FDOC); mmio_le_writel(control, spibar + PCH100_REG_FDOC);
return mmio_le_readl(spibar + PCH100_REG_FDOD); return mmio_le_readl(spibar + PCH100_REG_FDOD);
} else { default:
mmio_le_writel(control, spibar + ICH9_REG_FDOC); mmio_le_writel(control, spibar + ICH9_REG_FDOC);
return mmio_le_readl(spibar + ICH9_REG_FDOD); return mmio_le_readl(spibar + ICH9_REG_FDOD);
} }

116
ichspi.c
View File

@@ -29,6 +29,9 @@
#include "spi.h" #include "spi.h"
#include "ich_descriptors.h" #include "ich_descriptors.h"
/* Apollo Lake */
#define APL_REG_FREG12 0xe0 /* 32 Bytes Flash Region 12 */
/* Sunrise Point */ /* Sunrise Point */
/* Added HSFS Status bits */ /* Added HSFS Status bits */
@@ -1564,15 +1567,17 @@ static const char *const access_names[] = {
static enum ich_access_protection ich9_handle_frap(uint32_t frap, int i) static enum ich_access_protection ich9_handle_frap(uint32_t frap, int i)
{ {
const int rwperms_unknown = ARRAY_SIZE(access_names); const int rwperms_unknown = ARRAY_SIZE(access_names);
static const char *const region_names[5] = { static const char *const region_names[6] = {
"Flash Descriptor", "BIOS", "Management Engine", "Flash Descriptor", "BIOS", "Management Engine",
"Gigabit Ethernet", "Platform Data" "Gigabit Ethernet", "Platform Data", "Device Expansion",
}; };
const char *const region_name = i < ARRAY_SIZE(region_names) ? region_names[i] : "unknown"; const char *const region_name = i < ARRAY_SIZE(region_names) ? region_names[i] : "unknown";
uint32_t base, limit; uint32_t base, limit;
int rwperms; int rwperms;
int offset = ICH9_REG_FREG0 + i * 4; const int offset = i < 12
? ICH9_REG_FREG0 + i * 4
: APL_REG_FREG12 + (i - 12) * 4;
uint32_t freg = mmio_readl(ich_spibar + offset); uint32_t freg = mmio_readl(ich_spibar + offset);
if (i < 8) { if (i < 8) {
@@ -1716,8 +1721,10 @@ int ich_init_spi(void *spibar, enum ich_chipset ich_gen)
memset(&desc, 0x00, sizeof(struct ich_descriptors)); memset(&desc, 0x00, sizeof(struct ich_descriptors));
/* Moving registers / bits */ /* Moving registers / bits */
if (ich_generation == CHIPSET_100_SERIES_SUNRISE_POINT) { switch (ich_generation) {
num_freg = 10; case CHIPSET_100_SERIES_SUNRISE_POINT:
case CHIPSET_C620_SERIES_LEWISBURG:
case CHIPSET_APOLLO_LAKE:
num_pr = 6; /* Includes GPR0 */ num_pr = 6; /* Includes GPR0 */
reg_pr0 = PCH100_REG_FPR0; reg_pr0 = PCH100_REG_FPR0;
swseq_data.reg_ssfsc = PCH100_REG_SSFSC; swseq_data.reg_ssfsc = PCH100_REG_SSFSC;
@@ -1727,19 +1734,8 @@ int ich_init_spi(void *spibar, enum ich_chipset ich_gen)
hwseq_data.addr_mask = PCH100_FADDR_FLA; hwseq_data.addr_mask = PCH100_FADDR_FLA;
hwseq_data.only_4k = true; hwseq_data.only_4k = true;
hwseq_data.hsfc_fcycle = PCH100_HSFC_FCYCLE; hwseq_data.hsfc_fcycle = PCH100_HSFC_FCYCLE;
} else if (ich_generation == CHIPSET_C620_SERIES_LEWISBURG) { break;
num_freg = 12; /* 12 MMIO regs, but 16 regions in FD spec */ default:
num_pr = 6; /* Includes GPR0 */
reg_pr0 = PCH100_REG_FPR0;
swseq_data.reg_ssfsc = PCH100_REG_SSFSC;
swseq_data.reg_preop = PCH100_REG_PREOP;
swseq_data.reg_optype = PCH100_REG_OPTYPE;
swseq_data.reg_opmenu = PCH100_REG_OPMENU;
hwseq_data.addr_mask = PCH100_FADDR_FLA;
hwseq_data.only_4k = true;
hwseq_data.hsfc_fcycle = PCH100_HSFC_FCYCLE;
} else {
num_freg = 5;
num_pr = 5; num_pr = 5;
reg_pr0 = ICH9_REG_PR0; reg_pr0 = ICH9_REG_PR0;
swseq_data.reg_ssfsc = ICH9_REG_SSFS; swseq_data.reg_ssfsc = ICH9_REG_SSFS;
@@ -1749,6 +1745,21 @@ int ich_init_spi(void *spibar, enum ich_chipset ich_gen)
hwseq_data.addr_mask = ICH9_FADDR_FLA; hwseq_data.addr_mask = ICH9_FADDR_FLA;
hwseq_data.only_4k = false; hwseq_data.only_4k = false;
hwseq_data.hsfc_fcycle = HSFC_FCYCLE; hwseq_data.hsfc_fcycle = HSFC_FCYCLE;
break;
}
switch (ich_generation) {
case CHIPSET_100_SERIES_SUNRISE_POINT:
num_freg = 10;
break;
case CHIPSET_C620_SERIES_LEWISBURG:
num_freg = 12; /* 12 MMIO regs, but 16 regions in FD spec */
break;
case CHIPSET_APOLLO_LAKE:
num_freg = 16;
break;
default:
num_freg = 5;
break;
} }
switch (ich_generation) { switch (ich_generation) {
@@ -1834,10 +1845,16 @@ int ich_init_spi(void *spibar, enum ich_chipset ich_gen)
tmp = mmio_readl(ich_spibar + ICH9_REG_FADDR); tmp = mmio_readl(ich_spibar + ICH9_REG_FADDR);
msg_pdbg2("0x08: 0x%08x (FADDR)\n", tmp); msg_pdbg2("0x08: 0x%08x (FADDR)\n", tmp);
if (ich_gen == CHIPSET_100_SERIES_SUNRISE_POINT || ich_gen == CHIPSET_C620_SERIES_LEWISBURG) { switch (ich_gen) {
const uint32_t dlock = mmio_readl(ich_spibar + PCH100_REG_DLOCK); case CHIPSET_100_SERIES_SUNRISE_POINT:
msg_pdbg("0x0c: 0x%08x (DLOCK)\n", dlock); case CHIPSET_C620_SERIES_LEWISBURG:
prettyprint_pch100_reg_dlock(dlock); case CHIPSET_APOLLO_LAKE:
tmp = mmio_readl(ich_spibar + PCH100_REG_DLOCK);
msg_pdbg("0x0c: 0x%08x (DLOCK)\n", tmp);
prettyprint_pch100_reg_dlock(tmp);
break;
default:
break;
} }
if (desc_valid) { if (desc_valid) {
@@ -1898,37 +1915,51 @@ int ich_init_spi(void *spibar, enum ich_chipset ich_gen)
swseq_data.reg_opmenu, mmio_readl(ich_spibar + swseq_data.reg_opmenu)); swseq_data.reg_opmenu, mmio_readl(ich_spibar + swseq_data.reg_opmenu));
msg_pdbg("0x%zx: 0x%08x (OPMENU+4)\n", msg_pdbg("0x%zx: 0x%08x (OPMENU+4)\n",
swseq_data.reg_opmenu + 4, mmio_readl(ich_spibar + swseq_data.reg_opmenu + 4)); swseq_data.reg_opmenu + 4, mmio_readl(ich_spibar + swseq_data.reg_opmenu + 4));
if (ich_generation == CHIPSET_ICH8 && desc_valid) {
tmp = mmio_readl(ich_spibar + ICH8_REG_VSCC); if (desc_valid) {
msg_pdbg("0xC1: 0x%08x (VSCC)\n", tmp); switch (ich_gen) {
msg_pdbg("VSCC: "); case CHIPSET_ICH8:
prettyprint_ich_reg_vscc(tmp, FLASHROM_MSG_DEBUG, true); case CHIPSET_100_SERIES_SUNRISE_POINT:
} else if (ich_generation != CHIPSET_100_SERIES_SUNRISE_POINT && case CHIPSET_C620_SERIES_LEWISBURG:
ich_generation != CHIPSET_C620_SERIES_LEWISBURG) { case CHIPSET_APOLLO_LAKE:
if (ich_generation != CHIPSET_BAYTRAIL && desc_valid) { case CHIPSET_BAYTRAIL:
break;
default:
ichspi_bbar = mmio_readl(ich_spibar + ICH9_REG_BBAR); ichspi_bbar = mmio_readl(ich_spibar + ICH9_REG_BBAR);
msg_pdbg("0xA0: 0x%08x (BBAR)\n", msg_pdbg("0x%x: 0x%08x (BBAR)\n", ICH9_REG_BBAR, ichspi_bbar);
ichspi_bbar);
ich_set_bbar(0); ich_set_bbar(0);
break;
} }
if (desc_valid) { if (ich_gen == CHIPSET_ICH8) {
tmp = mmio_readl(ich_spibar + ICH8_REG_VSCC);
msg_pdbg("0x%x: 0x%08x (VSCC)\n", ICH8_REG_VSCC, tmp);
msg_pdbg("VSCC: ");
prettyprint_ich_reg_vscc(tmp, FLASHROM_MSG_DEBUG, true);
} else {
tmp = mmio_readl(ich_spibar + ICH9_REG_LVSCC); tmp = mmio_readl(ich_spibar + ICH9_REG_LVSCC);
msg_pdbg("0xC4: 0x%08x (LVSCC)\n", tmp); msg_pdbg("0x%x: 0x%08x (LVSCC)\n", ICH9_REG_LVSCC, tmp);
msg_pdbg("LVSCC: "); msg_pdbg("LVSCC: ");
prettyprint_ich_reg_vscc(tmp, FLASHROM_MSG_DEBUG, true); prettyprint_ich_reg_vscc(tmp, FLASHROM_MSG_DEBUG, true);
tmp = mmio_readl(ich_spibar + ICH9_REG_UVSCC); tmp = mmio_readl(ich_spibar + ICH9_REG_UVSCC);
msg_pdbg("0xC8: 0x%08x (UVSCC)\n", tmp); msg_pdbg("0x%x: 0x%08x (UVSCC)\n", ICH9_REG_UVSCC, tmp);
msg_pdbg("UVSCC: "); msg_pdbg("UVSCC: ");
prettyprint_ich_reg_vscc(tmp, FLASHROM_MSG_DEBUG, false); prettyprint_ich_reg_vscc(tmp, FLASHROM_MSG_DEBUG, false);
tmp = mmio_readl(ich_spibar + ICH9_REG_FPB);
msg_pdbg("0xD0: 0x%08x (FPB)\n", tmp);
} }
}
if (desc_valid) { switch (ich_gen) {
case CHIPSET_ICH8:
case CHIPSET_100_SERIES_SUNRISE_POINT:
case CHIPSET_C620_SERIES_LEWISBURG:
case CHIPSET_APOLLO_LAKE:
break;
default:
tmp = mmio_readl(ich_spibar + ICH9_REG_FPB);
msg_pdbg("0x%x: 0x%08x (FPB)\n", ICH9_REG_FPB, tmp);
break;
}
if (read_ich_descriptors_via_fdo(ich_gen, ich_spibar, &desc) == ICH_RET_OK) if (read_ich_descriptors_via_fdo(ich_gen, ich_spibar, &desc) == ICH_RET_OK)
prettyprint_ich_descriptors(ich_gen, &desc); prettyprint_ich_descriptors(ich_gen, &desc);
@@ -1955,6 +1986,11 @@ int ich_init_spi(void *spibar, enum ich_chipset ich_gen)
ich_spi_mode = ich_hwseq; ich_spi_mode = ich_hwseq;
} }
if (ich_spi_mode == ich_auto && ich_gen == CHIPSET_APOLLO_LAKE) {
msg_pdbg("Enabling hardware sequencing by default for Apollo Lake.\n");
ich_spi_mode = ich_hwseq;
}
if (ich_spi_mode == ich_hwseq) { if (ich_spi_mode == ich_hwseq) {
if (!desc_valid) { if (!desc_valid) {
msg_perr("Hardware sequencing was requested " msg_perr("Hardware sequencing was requested "

View File

@@ -126,6 +126,7 @@ static void usage(char *argv[], char *error)
"\t- \"ich9\",\n" "\t- \"ich9\",\n"
"\t- \"ich10\",\n" "\t- \"ich10\",\n"
"\t- \"silvermont\" for chipsets from Intel's Silvermont architecture (e.g. Bay Trail),\n" "\t- \"silvermont\" for chipsets from Intel's Silvermont architecture (e.g. Bay Trail),\n"
"\t- \"apollo\" for Intel's Apollo Lake SoC.\n"
"\t- \"5\" or \"ibex\" for Intel's 5 series chipsets,\n" "\t- \"5\" or \"ibex\" for Intel's 5 series chipsets,\n"
"\t- \"6\" or \"cougar\" for Intel's 6 series chipsets,\n" "\t- \"6\" or \"cougar\" for Intel's 6 series chipsets,\n"
"\t- \"7\" or \"panther\" for Intel's 7 series chipsets.\n" "\t- \"7\" or \"panther\" for Intel's 7 series chipsets.\n"
@@ -220,6 +221,8 @@ int main(int argc, char *argv[])
else if ((strcmp(csn, "100") == 0) || else if ((strcmp(csn, "100") == 0) ||
(strcmp(csn, "sunrise") == 0)) (strcmp(csn, "sunrise") == 0))
cs = CHIPSET_100_SERIES_SUNRISE_POINT; cs = CHIPSET_100_SERIES_SUNRISE_POINT;
else if (strcmp(csn, "apollo") == 0)
cs = CHIPSET_APOLLO_LAKE;
} }
ret = read_ich_descriptors_from_dump(buf, len, &cs, &desc); ret = read_ich_descriptors_from_dump(buf, len, &cs, &desc);