2015-05-13 03:19:47 +02:00
|
|
|
/*
|
|
|
|
* This file is part of the coreboot project.
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* 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 <device/device.h>
|
|
|
|
#include <device/pci.h>
|
|
|
|
#include <pc80/isa-dma.h>
|
|
|
|
#include <pc80/i8259.h>
|
|
|
|
#include <arch/ioapic.h>
|
2017-04-12 01:05:23 +02:00
|
|
|
#include <intelblocks/itss.h>
|
2017-09-29 02:06:01 +02:00
|
|
|
#include <intelblocks/lpc_lib.h>
|
2017-03-08 13:29:40 +01:00
|
|
|
#include <intelblocks/pcr.h>
|
2015-05-13 03:19:47 +02:00
|
|
|
#include <reg_script.h>
|
|
|
|
#include <soc/iomap.h>
|
2017-03-08 13:29:40 +01:00
|
|
|
#include <soc/pcr_ids.h>
|
2017-12-08 17:51:21 +01:00
|
|
|
#include <soc/intel/common/block/lpc/lpc_def.h>
|
2017-09-29 02:06:01 +02:00
|
|
|
|
2019-03-21 11:10:03 +01:00
|
|
|
#include "chip.h"
|
|
|
|
|
2017-09-29 02:06:01 +02:00
|
|
|
/**
|
|
|
|
PCH preserved MMIO range, 24 MB, from 0xFD000000 to 0xFE7FFFFF
|
|
|
|
**/
|
|
|
|
|
|
|
|
static const struct lpc_mmio_range skl_lpc_fixed_mmio_ranges[] = {
|
|
|
|
{ PCH_PRESERVED_BASE_ADDRESS, PCH_PRESERVED_BASE_SIZE },
|
|
|
|
{ 0, 0 }
|
|
|
|
};
|
|
|
|
|
|
|
|
const struct lpc_mmio_range *soc_get_fixed_mmio_ranges(void)
|
|
|
|
{
|
|
|
|
return skl_lpc_fixed_mmio_ranges;
|
|
|
|
}
|
2015-05-13 03:19:47 +02:00
|
|
|
|
|
|
|
static void pch_enable_ioapic(struct device *dev)
|
|
|
|
{
|
|
|
|
u32 reg32;
|
2015-08-13 16:05:22 +02:00
|
|
|
/* PCH-LP has 120 redirection entries */
|
|
|
|
const int redir_entries = 120;
|
2015-05-13 03:19:47 +02:00
|
|
|
|
2015-05-13 03:23:27 +02:00
|
|
|
set_ioapic_id((void *)IO_APIC_ADDR, 0x02);
|
2015-05-13 03:19:47 +02:00
|
|
|
|
|
|
|
/* affirm full set of redirection table entries ("write once") */
|
2015-05-13 03:23:27 +02:00
|
|
|
reg32 = io_apic_read((void *)IO_APIC_ADDR, 0x01);
|
2015-05-13 03:19:47 +02:00
|
|
|
|
|
|
|
reg32 &= ~0x00ff0000;
|
2015-08-13 16:05:22 +02:00
|
|
|
reg32 |= (redir_entries - 1) << 16;
|
2015-05-13 03:19:47 +02:00
|
|
|
|
2015-05-13 03:23:27 +02:00
|
|
|
io_apic_write((void *)IO_APIC_ADDR, 0x01, reg32);
|
2015-05-13 03:19:47 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Select Boot Configuration register (0x03) and
|
|
|
|
* use Processor System Bus (0x01) to deliver interrupts.
|
|
|
|
*/
|
2015-05-13 03:23:27 +02:00
|
|
|
io_apic_write((void *)IO_APIC_ADDR, 0x03, 0x01);
|
2015-05-13 03:19:47 +02:00
|
|
|
}
|
|
|
|
|
2017-09-29 02:06:01 +02:00
|
|
|
void soc_get_gen_io_dec_range(const struct device *dev, uint32_t *gen_io_dec)
|
2015-05-13 03:19:47 +02:00
|
|
|
{
|
2019-07-13 21:16:25 +02:00
|
|
|
const config_t *config = config_of(dev);
|
2015-05-13 03:19:47 +02:00
|
|
|
|
2017-09-29 02:06:01 +02:00
|
|
|
gen_io_dec[0] = config->gen1_dec;
|
|
|
|
gen_io_dec[1] = config->gen2_dec;
|
|
|
|
gen_io_dec[2] = config->gen3_dec;
|
|
|
|
gen_io_dec[3] = config->gen4_dec;
|
2015-05-13 03:19:47 +02:00
|
|
|
}
|
|
|
|
|
2017-09-29 02:06:01 +02:00
|
|
|
void soc_setup_dmi_pcr_io_dec(uint32_t *gen_io_dec)
|
|
|
|
{
|
|
|
|
/* Mirror these same settings in DMI PCR */
|
|
|
|
pcr_write32(PID_DMI, PCR_DMI_LPCLGIR1, gen_io_dec[0]);
|
|
|
|
pcr_write32(PID_DMI, PCR_DMI_LPCLGIR2, gen_io_dec[1]);
|
|
|
|
pcr_write32(PID_DMI, PCR_DMI_LPCLGIR3, gen_io_dec[2]);
|
|
|
|
pcr_write32(PID_DMI, PCR_DMI_LPCLGIR4, gen_io_dec[3]);
|
|
|
|
}
|
2015-05-13 03:19:47 +02:00
|
|
|
|
|
|
|
static const struct reg_script pch_misc_init_script[] = {
|
|
|
|
/* Setup NMI on errors, disable SERR */
|
|
|
|
REG_IO_RMW8(0x61, ~0xf0, (1 << 2)),
|
|
|
|
/* Disable NMI sources */
|
|
|
|
REG_IO_OR8(0x70, (1 << 7)),
|
|
|
|
/* Enable BIOS updates outside of SMM */
|
|
|
|
REG_PCI_RMW8(0xdc, ~(1 << 5), 0),
|
|
|
|
REG_SCRIPT_END
|
|
|
|
};
|
|
|
|
|
2018-01-10 06:21:50 +01:00
|
|
|
void lpc_soc_init(struct device *dev)
|
2015-05-13 03:19:47 +02:00
|
|
|
{
|
2019-07-13 21:16:25 +02:00
|
|
|
const config_t *const config = config_of(dev);
|
2019-02-23 19:24:51 +01:00
|
|
|
|
2015-05-13 03:19:47 +02:00
|
|
|
/* Legacy initialization */
|
|
|
|
isa_dma_init();
|
2017-09-29 02:06:01 +02:00
|
|
|
reg_script_run_on_dev(PCH_DEV_LPC, pch_misc_init_script);
|
|
|
|
|
|
|
|
/* Enable CLKRUN_EN for power gating LPC */
|
|
|
|
lpc_enable_pci_clk_cntl();
|
|
|
|
|
|
|
|
/* Set LPC Serial IRQ mode */
|
2019-02-23 19:24:51 +01:00
|
|
|
lpc_set_serirq_mode(config->serirq_mode);
|
2015-05-13 03:19:47 +02:00
|
|
|
|
|
|
|
/* Interrupt configuration */
|
|
|
|
pch_enable_ioapic(dev);
|
2017-09-29 02:06:01 +02:00
|
|
|
soc_pch_pirq_init(dev);
|
2015-05-13 03:19:47 +02:00
|
|
|
setup_i8259();
|
|
|
|
i8259_configure_irq_trigger(9, 1);
|
|
|
|
}
|