blob: d24416fd6c64425b8efe134cd4220101c2b17bd3 [file] [log] [blame]
Lance Zhaof51b1272015-11-09 17:06:34 -08001/*
2 * This file is part of the coreboot project.
3 *
4 * Copyright (C) 2016 Intel Corp.
Lance Zhaoe904c7c2015-11-10 19:00:18 -08005 * (Written by Lance Zhao <lijian.zhao@intel.com> for Intel Corp.)
Lance Zhaof51b1272015-11-09 17:06:34 -08006 *
7 * This program is free software; you can redistribute it and/or modify
8 * it under the terms of the GNU General Public License as published by
9 * the Free Software Foundation; either version 2 of the License, or
10 * (at your option) any later version.
Martin Rothebabfad2016-04-10 11:09:16 -060011 *
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU General Public License for more details.
Lance Zhaof51b1272015-11-09 17:06:34 -080016 */
17
18#include <arch/acpi.h>
Lance Zhao1bd0c0c2016-04-19 18:04:21 -070019#include <arch/acpigen.h>
Lance Zhao2fc82d62015-11-16 18:33:21 -080020#include <arch/ioapic.h>
21#include <arch/smp/mpspec.h>
Lance Zhao1bd0c0c2016-04-19 18:04:21 -070022#include <cbmem.h>
Lance Zhaoe904c7c2015-11-10 19:00:18 -080023#include <cpu/x86/smm.h>
Aaron Durbin1ee6f0b2016-06-10 15:50:34 -050024#include <cpu/cpu.h>
Lance Zhaoe904c7c2015-11-10 19:00:18 -080025#include <soc/acpi.h>
Hannah Williams0f61da82016-04-18 13:47:08 -070026#include <soc/intel/common/acpi.h>
Lance Zhaoe904c7c2015-11-10 19:00:18 -080027#include <soc/iomap.h>
28#include <soc/pm.h>
Lance Zhao1bd0c0c2016-04-19 18:04:21 -070029#include <soc/nvs.h>
Lance Zhaof51b1272015-11-09 17:06:34 -080030
Hannah Williams0f61da82016-04-18 13:47:08 -070031#define CSTATE_RES(address_space, width, offset, address) \
32 { \
33 .space_id = address_space, \
34 .bit_width = width, \
35 .bit_offset = offset, \
36 .addrl = address, \
37 }
38
Lance Zhaof51b1272015-11-09 17:06:34 -080039unsigned long acpi_fill_mcfg(unsigned long current)
40{
Lance Zhao2c34e312015-11-16 18:13:23 -080041 /* PCI Segment Group 0, Start Bus Number 0, End Bus Number is 255 */
42 current += acpi_create_mcfg_mmconfig((void *) current,
43 CONFIG_MMCONF_BASE_ADDRESS, 0, 0,
44 255);
Lance Zhaoe904c7c2015-11-10 19:00:18 -080045 return current;
Lance Zhaof51b1272015-11-09 17:06:34 -080046}
Lance Zhaoe904c7c2015-11-10 19:00:18 -080047
Lance Zhaoe904c7c2015-11-10 19:00:18 -080048static int acpi_sci_irq(void)
49{
50 int sci_irq = 9;
51 return sci_irq;
52}
53
Lance Zhao2fc82d62015-11-16 18:33:21 -080054static unsigned long acpi_madt_irq_overrides(unsigned long current)
55{
56 int sci = acpi_sci_irq();
57 uint16_t flags = MP_IRQ_TRIGGER_LEVEL | MP_IRQ_POLARITY_LOW;;
58
59 /* INT_SRC_OVR */
60 current += acpi_create_madt_irqoverride((void *)current, 0, 0, 2, 0);
61
62 /* SCI */
63 current += acpi_create_madt_irqoverride((void *)current, 0, sci, sci, flags);
64
65 return current;
66}
67
68unsigned long acpi_fill_madt(unsigned long current)
69{
70 /* Local APICs */
71 current = acpi_create_madt_lapics(current);
72
73 /* IOAPIC */
74 current += acpi_create_madt_ioapic((void *) current,
75 2, IO_APIC_ADDR, 0);
76
77 return acpi_madt_irq_overrides(current);
78}
79
Aaron Durbinc3ee3f62016-05-11 10:35:49 -050080void acpi_fill_fadt(acpi_fadt_t * fadt)
Lance Zhaoe904c7c2015-11-10 19:00:18 -080081{
82 const uint16_t pmbase = ACPI_PMIO_BASE;
83
Aaron Durbinc3ee3f62016-05-11 10:35:49 -050084 /* Use ACPI 5.0 revision. */
85 fadt->header.revision = ACPI_FADT_REV_ACPI_5_0;
86
Lance Zhaoe904c7c2015-11-10 19:00:18 -080087 fadt->sci_int = acpi_sci_irq();
88 fadt->smi_cmd = 0; /* No Smi Handler as SMI_CMD is 0*/
89
90 fadt->pm1a_evt_blk = pmbase + PM1_STS;
91 fadt->pm1a_cnt_blk = pmbase + PM1_CNT;
92 fadt->pm_tmr_blk = pmbase + PM1_TMR;
93 fadt->gpe0_blk = pmbase + GPE0_STS(0);
94
95 fadt->pm1_evt_len = 4;
96 fadt->pm1_cnt_len = 2;
97 fadt->pm_tmr_len = 4;
98 /* There are 4 GPE0 STS/EN pairs each 32 bits wide. */
99 fadt->gpe0_blk_len = 2 * GPE0_REG_MAX * sizeof(uint32_t);
100 fadt->p_lvl2_lat = ACPI_FADT_C2_NOT_SUPPORTED;
101 fadt->p_lvl3_lat = ACPI_FADT_C3_NOT_SUPPORTED;
102 fadt->flush_size = 0x400; /* twice of cache size*/
103 fadt->flush_stride = 0x10; /* Cache line width */
104 fadt->duty_offset = 1;
105 fadt->duty_width = 3;
106 fadt->day_alrm = 0xd;
107 fadt->iapc_boot_arch = ACPI_FADT_LEGACY_DEVICES | ACPI_FADT_8042;
108
109 fadt->flags = ACPI_FADT_WBINVD | ACPI_FADT_C1_SUPPORTED |
110 ACPI_FADT_C2_MP_SUPPORTED | ACPI_FADT_SLEEP_BUTTON |
111 ACPI_FADT_RESET_REGISTER | ACPI_FADT_SEALED_CASE |
112 ACPI_FADT_S4_RTC_WAKE | ACPI_FADT_PLATFORM_CLOCK;
113
114 fadt->reset_reg.space_id = 1;
115 fadt->reset_reg.bit_width = 8;
116 fadt->reset_reg.addrl = 0xcf9;
117 fadt->reset_value = 6;
118
119 fadt->x_pm1a_evt_blk.space_id = 1;
120 fadt->x_pm1a_evt_blk.bit_width = fadt->pm1_evt_len * 8;
121 fadt->x_pm1a_evt_blk.addrl = pmbase + PM1_STS;
122
123 fadt->x_pm1b_evt_blk.space_id = 1;
124
125 fadt->x_pm1a_cnt_blk.space_id = 1;
126 fadt->x_pm1a_cnt_blk.bit_width = fadt->pm1_cnt_len * 8;
127 fadt->x_pm1a_cnt_blk.addrl = pmbase + PM1_CNT;
128
129 fadt->x_pm1b_cnt_blk.space_id = 1;
130
131 fadt->x_pm_tmr_blk.space_id = 1;
132 fadt->x_pm_tmr_blk.bit_width = fadt->pm_tmr_len * 8;
133 fadt->x_pm_tmr_blk.addrl = pmbase + PM1_TMR;
134
135 fadt->x_gpe1_blk.space_id = 1;
Lance Zhaof51b1272015-11-09 17:06:34 -0800136}
Zhao, Lijian30461a92015-12-01 09:14:20 -0800137
138unsigned long southbridge_write_acpi_tables(device_t device,
139 unsigned long current,
140 struct acpi_rsdp *rsdp)
141{
142 return acpi_write_hpet(device, current, rsdp);
143}
Lance Zhao1bd0c0c2016-04-19 18:04:21 -0700144
145static void acpi_create_gnvs(struct global_nvs_t *gnvs)
146{
Furquan Shaikhd01f5a02016-06-13 22:23:49 -0700147 if (IS_ENABLED(CONFIG_CONSOLE_CBMEM))
148 gnvs->cbmc = (uintptr_t)cbmem_find(CBMEM_ID_CONSOLE);
149
Lance Zhao1bd0c0c2016-04-19 18:04:21 -0700150 if (IS_ENABLED(CONFIG_CHROMEOS)) {
151 /* Initialize Verified Boot data */
152 chromeos_init_vboot(&gnvs->chromeos);
153 gnvs->chromeos.vbt2 = ACTIVE_ECFW_RO;
154 }
155}
156
157void southbridge_inject_dsdt(device_t device)
158{
159 struct global_nvs_t *gnvs;
160
161 gnvs = cbmem_find(CBMEM_ID_ACPI_GNVS);
162
163 if (gnvs) {
164 acpi_create_gnvs(gnvs);
165 acpi_save_gnvs((uintptr_t)gnvs);
Aaron Durbin1ee6f0b2016-06-10 15:50:34 -0500166 /* And tell SMI about it */
167 smm_setup_structures(gnvs, NULL, NULL);
Lance Zhao1bd0c0c2016-04-19 18:04:21 -0700168
169 /* Add it to DSDT. */
170 acpigen_write_scope("\\");
171 acpigen_write_name_dword("NVSA", (uintptr_t)gnvs);
172 acpigen_pop_len();
173 }
174}
Hannah Williams0f61da82016-04-18 13:47:08 -0700175static acpi_cstate_t cstate_map[] = {
176 {
177 /* C1 */
178 .ctype = 1, /* ACPI C1 */
179 .latency = 1,
180 .power = 1000,
181 .resource = CSTATE_RES(ACPI_ADDRESS_SPACE_FIXED, 0, 0, 0),
182 },
183 {
184 .ctype = 2, /* ACPI C2 */
185 .latency = 50,
186 .power = 10,
187 .resource = CSTATE_RES(ACPI_ADDRESS_SPACE_IO, 8, 0, 0x415),
188 },
189 {
190 .ctype = 3, /* ACPI C3 */
191 .latency = 150,
192 .power = 10,
193 .resource = CSTATE_RES(ACPI_ADDRESS_SPACE_IO, 8, 0, 0x419),
194 }
195};
196
197acpi_cstate_t *soc_get_cstate_map(int *entries)
198{
199 *entries = ARRAY_SIZE(cstate_map);
200 return cstate_map;
201}
202
203uint16_t soc_get_acpi_base_address(void)
204{
205 return ACPI_PMIO_BASE;
206}