2017-11-29 11:47:13 +01:00
|
|
|
/*
|
|
|
|
* This file is part of the coreboot project.
|
|
|
|
*
|
|
|
|
* Copyright (C) 2017 Intel Corp.
|
|
|
|
*
|
|
|
|
* 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 <arch/io.h>
|
2019-03-01 12:43:02 +01:00
|
|
|
#include <device/pci_ops.h>
|
2017-11-29 11:47:13 +01:00
|
|
|
#include <chip.h>
|
|
|
|
#include <console/console.h>
|
|
|
|
#include <device/device.h>
|
|
|
|
#include <device/pci.h>
|
|
|
|
#include <soc/iomap.h>
|
|
|
|
#include <soc/pci_devs.h>
|
|
|
|
#include <soc/thermal.h>
|
|
|
|
|
|
|
|
#define MAX_TRIP_TEMP 205
|
|
|
|
#define DEFAULT_TRIP_TEMP 50
|
|
|
|
|
|
|
|
static void *pch_thermal_get_bar(struct device *dev)
|
|
|
|
{
|
|
|
|
uintptr_t bar;
|
|
|
|
|
|
|
|
bar = pci_read_config32(dev, PCI_BASE_ADDRESS_0);
|
|
|
|
/*
|
|
|
|
* Bits [31:12] are the base address as per EDS for Thermal Device,
|
|
|
|
* Don't care about [11:0] bits
|
|
|
|
*/
|
|
|
|
return (void *)(bar & ~PCI_BASE_ADDRESS_MEM_ATTR_MASK);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void pch_thermal_set_bar(struct device *dev, uintptr_t tempbar)
|
|
|
|
{
|
|
|
|
uint8_t pcireg;
|
|
|
|
|
|
|
|
/* Assign Resources to Thermal Device */
|
|
|
|
/* Clear BIT 1-2 of Command Register */
|
|
|
|
pcireg = pci_read_config8(dev, PCI_COMMAND);
|
|
|
|
pcireg &= ~(PCI_COMMAND_MASTER | PCI_COMMAND_MEMORY);
|
|
|
|
pci_write_config8(dev, PCI_COMMAND, pcireg);
|
|
|
|
|
|
|
|
/* Program Temporary BAR for Thermal Device */
|
|
|
|
pci_write_config32(dev, PCI_BASE_ADDRESS_0, tempbar);
|
|
|
|
pci_write_config32(dev, PCI_BASE_ADDRESS_1, 0x0);
|
|
|
|
|
|
|
|
/* Enable Bus Master and MMIO Space */
|
|
|
|
pcireg = pci_read_config8(dev, PCI_COMMAND);
|
|
|
|
pcireg |= PCI_COMMAND_MASTER | PCI_COMMAND_MEMORY;
|
|
|
|
pci_write_config8(dev, PCI_COMMAND, pcireg);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* PCH Low Temp Threshold (LTT) */
|
|
|
|
static uint16_t pch_get_ltt_value(struct device *dev)
|
|
|
|
{
|
|
|
|
static struct soc_intel_skylake_config *config;
|
|
|
|
uint16_t ltt_value;
|
|
|
|
uint16_t trip_temp = DEFAULT_TRIP_TEMP;
|
|
|
|
|
|
|
|
config = dev->chip_info;
|
|
|
|
|
|
|
|
if (config->pch_trip_temp)
|
|
|
|
trip_temp = config->pch_trip_temp;
|
|
|
|
|
|
|
|
if (trip_temp > MAX_TRIP_TEMP)
|
|
|
|
die("Input PCH temp trip is higher than allowed range!");
|
|
|
|
|
|
|
|
/* Trip Point Temp = (LTT / 2 - 50 degree C) */
|
|
|
|
ltt_value = (trip_temp + 50) * 2;
|
|
|
|
|
|
|
|
return ltt_value;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Enable thermal sensor power management */
|
|
|
|
void pch_thermal_configuration(void)
|
|
|
|
{
|
|
|
|
uint16_t reg16;
|
|
|
|
struct device *dev = PCH_DEV_THERMAL;
|
2018-10-06 10:28:33 +02:00
|
|
|
if (!dev) {
|
|
|
|
printk(BIOS_ERR, "PCH_DEV_THERMAL device not found!\n");
|
|
|
|
return;
|
|
|
|
}
|
2017-11-29 11:47:13 +01:00
|
|
|
void *thermalbar = pch_thermal_get_bar(dev);
|
|
|
|
|
|
|
|
/* Use default pre-ram bar */
|
|
|
|
if (!thermalbar) {
|
|
|
|
pch_thermal_set_bar(dev, THERMAL_BASE_ADDRESS);
|
|
|
|
thermalbar = (void *)THERMAL_BASE_ADDRESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Set Low Temp Threshold (LTT) at TSPM offset 0x1c[8:0] */
|
|
|
|
reg16 = read16(thermalbar + THERMAL_SENSOR_POWER_MANAGEMENT);
|
|
|
|
reg16 &= ~0x1ff;
|
|
|
|
/* Low Temp Threshold (LTT) */
|
|
|
|
reg16 |= pch_get_ltt_value(dev);
|
|
|
|
write16(thermalbar + THERMAL_SENSOR_POWER_MANAGEMENT, reg16);
|
|
|
|
}
|