1 /* SPDX-License-Identifier: GPL-2.0-only */
4 #include <acpi/acpigen.h>
7 #include <console/console.h>
10 #include <device/device.h>
11 #include <device/dram/spd.h>
12 #include <device/pci_ids.h>
13 #include <drivers/intel/gma/opregion.h>
15 #include <intelblocks/gpio.h>
16 #include <intelblocks/pmclib.h>
27 const char *mainboard_vbt_filename(void)
29 const struct eeprom_bmc_settings
*bmc_cfg
= get_bmc_settings();
31 if (bmc_cfg
&& bmc_cfg
->efp3_displayport
)
32 return "vbt-avalanche.bin";
34 return "vbt.bin"; /* Poseidon */
37 /* FIXME: Example code below */
39 static void mb_configure_dp1_pwr(bool enable
)
41 gpio_output(GPP_K3
, enable
);
44 static void mb_configure_dp2_pwr(bool enable
)
46 gpio_output(GPP_K4
, enable
);
49 static void mb_configure_dp3_pwr(bool enable
)
51 gpio_output(GPP_K5
, enable
);
54 static void mb_hda_amp_enable(bool enable
)
56 gpio_output(GPP_C19
, enable
);
59 static void mb_usb31_rp1_pwr_enable(bool enable
)
61 gpio_output(GPP_G0
, enable
);
64 static void mb_usb31_rp2_pwr_enable(bool enable
)
66 gpio_output(GPP_G1
, enable
);
69 static void mb_usb31_fp_pwr_enable(bool enable
)
71 gpio_output(GPP_G2
, enable
);
74 static void mb_usb2_fp1_pwr_enable(bool enable
)
76 gpio_output(GPP_G3
, enable
);
79 static void mb_usb2_fp2_pwr_enable(bool enable
)
81 gpio_output(GPP_G4
, enable
);
84 static void copy_meminfo(const struct dimm_info
*dimm
, union eeprom_dimm_layout
*l
)
86 memset(l
, 0, sizeof(*l
));
87 if (dimm
->dimm_size
== 0)
90 strncpy(l
->name
, (char *)dimm
->module_part_number
, sizeof(l
->name
) - 1);
91 l
->capacity_mib
= dimm
->dimm_size
;
92 l
->data_width_bits
= 8 * (1 << (dimm
->bus_width
& 0x7));
93 l
->bus_width_bits
= l
->data_width_bits
+ 8 * ((dimm
->bus_width
>> 3) & 0x3);
94 l
->ranks
= dimm
->rank_per_dimm
;
96 strncpy(l
->manufacturer
, spd_manufacturer_name(dimm
->mod_id
),
97 sizeof(l
->manufacturer
) - 1);
101 * Collect board specific settings and update the CFG EEPROM if necessary.
102 * This allows the BMC webui to display the current hardware configuration.
104 static void update_board_layout(void)
106 struct eeprom_board_layout layout
= {0};
108 printk(BIOS_INFO
, "MB: Collecting Board Layout information\n");
110 /* Update CPU fields */
111 for (struct device
*cpu
= all_devices
; cpu
; cpu
= cpu
->next
) {
112 if (!is_enabled_cpu(cpu
))
115 if (!layout
.cpu_name
[0])
116 strncpy(layout
.cpu_name
, cpu
->name
, sizeof(layout
.cpu_name
));
119 if (cpuid_get_max_func() >= 0x16)
120 layout
.cpu_max_non_turbo_frequency
= cpuid_eax(0x16);
123 strcpy(layout
.pch_name
, "Cannonlake-H C246");
126 struct memory_info
*meminfo
= cbmem_find(CBMEM_ID_MEMINFO
);
128 const size_t meminfo_max
= MIN(meminfo
->dimm_cnt
, ARRAY_SIZE(meminfo
->dimm
));
129 for (size_t i
= 0; i
< MIN(meminfo_max
, ARRAY_SIZE(layout
.dimm
)); i
++)
130 copy_meminfo(&meminfo
->dimm
[i
], &layout
.dimm
[i
]);
134 layout
.signature
= CRC(layout
.raw_layout
, sizeof(layout
.raw_layout
), crc32_byte
);
136 printk(BIOS_DEBUG
, "BOARD LAYOUT:\n");
137 printk(BIOS_DEBUG
, " Signature : 0x%x\n", layout
.signature
);
138 printk(BIOS_DEBUG
, " CPU name : %s\n", layout
.cpu_name
);
139 printk(BIOS_DEBUG
, " CPU count : %u\n", layout
.cpu_count
);
140 printk(BIOS_DEBUG
, " CPU freq : %u\n", layout
.cpu_max_non_turbo_frequency
);
141 printk(BIOS_DEBUG
, " PCH name : %s\n", layout
.pch_name
);
142 for (size_t i
= 0; i
< ARRAY_SIZE(layout
.dimm
); i
++)
143 printk(BIOS_DEBUG
, " DRAM SIZE : %u\n", layout
.dimm
[i
].capacity_mib
);
145 if (write_board_settings(&layout
))
146 printk(BIOS_ERR
, "MB: Failed to update Board Layout\n");
149 static void mainboard_init(void *chip_info
)
151 /* Enable internal speaker amplifier */
152 if (get_board_settings()->front_panel_audio
== 2)
153 mb_hda_amp_enable(1);
155 mb_hda_amp_enable(0);
158 static void mainboard_final(struct device
*dev
)
160 update_board_layout();
162 /* Encoding: 0 -> S0, 1 -> S5 */
163 const bool on
= !get_board_settings()->power_state_after_g3
;
165 pmc_soc_set_afterg3_en(on
);
168 static const char *format_pn(const char *prefix
, size_t offset
)
170 static char buffer
[32 + HERMES_SN_PN_LENGTH
] = { 0 };
172 const char *part_num
= eeprom_read_serial(offset
, "N/A");
174 snprintf(buffer
, sizeof(buffer
), "%s%s", prefix
, part_num
);
179 static void mainboard_smbios_strings(struct device
*dev
, struct smbios_type11
*t
)
181 const size_t board_offset
= offsetof(struct eeprom_layout
, board_part_number
);
182 const size_t product_offset
= offsetof(struct eeprom_layout
, product_part_number
);
183 t
->count
= smbios_add_string(t
->eos
, format_pn("Board P/N: ", board_offset
));
184 t
->count
= smbios_add_string(t
->eos
, format_pn("Product P/N: ", product_offset
));
187 #if CONFIG(HAVE_ACPI_TABLES)
188 static void mainboard_acpi_fill_ssdt(const struct device
*dev
)
190 const struct eeprom_board_settings
*const board_cfg
= get_board_settings();
192 const unsigned int usb_power_gpios
[] = { GPP_G0
, GPP_G1
, GPP_G2
, GPP_G3
, GPP_G4
};
194 /* Function pointer to write STXS or CTXS according to EEPROM board setting */
195 int (*acpigen_write_soc_gpio_op
)(unsigned int gpio_num
);
197 if (board_cfg
->usb_powered_in_s5
)
198 acpigen_write_soc_gpio_op
= acpigen_soc_set_tx_gpio
;
200 acpigen_write_soc_gpio_op
= acpigen_soc_clear_tx_gpio
;
202 acpigen_write_method("\\_SB.MPTS", 1);
204 acpigen_write_if_lequal_op_int(ARG0_OP
, 5);
206 for (size_t i
= 0; i
< ARRAY_SIZE(usb_power_gpios
); i
++)
207 acpigen_write_soc_gpio_op(usb_power_gpios
[i
]);
211 if (!board_cfg
->wake_on_usb
) {
213 acpigen_emit_byte(LNOT_OP
);
214 acpigen_emit_byte(LLESS_OP
);
215 acpigen_emit_byte(ARG0_OP
);
216 acpigen_write_integer(ACPI_S3
);
218 acpigen_write_store_int_to_namestr(0, "\\_SB.PCI0.XHCI.PMEE");
227 static void mainboard_enable(struct device
*dev
)
229 /* FIXME: Do runtime configuration once the board is production ready */
230 mb_configure_dp1_pwr(1);
231 mb_configure_dp2_pwr(1);
232 mb_configure_dp3_pwr(1);
234 mb_usb31_rp1_pwr_enable(1);
235 mb_usb31_rp2_pwr_enable(1);
236 mb_usb31_fp_pwr_enable(1);
237 mb_usb2_fp1_pwr_enable(1);
238 mb_usb2_fp2_pwr_enable(1);
240 dev
->ops
->final
= mainboard_final
;
241 dev
->ops
->get_smbios_strings
= mainboard_smbios_strings
;
243 #if CONFIG(HAVE_ACPI_TABLES)
244 dev
->ops
->acpi_fill_ssdt
= mainboard_acpi_fill_ssdt
;
248 struct chip_operations mainboard_ops
= {
249 .init
= mainboard_init
,
250 .enable_dev
= mainboard_enable
,
253 static void log_reset_causes(void)
255 struct chipset_power_state
*ps
= pmc_get_power_state();
258 printk(BIOS_ERR
, "chipset_power_state not found!\n");
263 struct eeprom_reset_cause_regs regs
;
264 uint8_t raw
[sizeof(struct eeprom_reset_cause_regs
)];
267 .gblrst_cause0
= ps
->gblrst_cause
[0],
268 .gblrst_cause1
= ps
->gblrst_cause
[1],
269 .hpr_cause0
= ps
->hpr_cause0
,
273 const size_t base
= offsetof(struct eeprom_layout
, reset_cause_regs
);
274 for (size_t i
= 0; i
< ARRAY_SIZE(reset_cause
.raw
); i
++)
275 eeprom_write_byte(reset_cause
.raw
[i
], base
+ i
);
278 /* Must happen before MPinit */
279 static void mainboard_early(void *unused
)
281 const struct eeprom_board_settings
*const board_cfg
= get_board_settings();
282 config_t
*config
= config_of_soc();
285 config
->deep_s5_enable_ac
= board_cfg
->deep_sx_enabled
;
286 config
->deep_s5_enable_dc
= board_cfg
->deep_sx_enabled
;
288 config
->disable_vmx
= board_cfg
->vtx_disabled
;
290 if (check_signature(offsetof(struct eeprom_layout
, supd
), FSPS_UPD_SIGNATURE
)) {
297 READ_EEPROM_FSP_S((&supd
), FspsConfig
.TurboMode
);
298 config
->cpu_turbo_disable
= !supd
.FspsConfig
.TurboMode
;
304 BOOT_STATE_INIT_ENTRY(BS_PRE_DEVICE
, BS_ON_EXIT
, mainboard_early
, NULL
);
307 * coreboot only exposes the last framebuffer that is set up.
308 * The ASPEED BMC will always be initialized after the IGD due to its higher
309 * bus number. To have coreboot only expose the IGD framebuffer skip the init
310 * function on the ASPEED.
312 static void mainboard_configure_internal_gfx(void *unused
)
316 if (get_board_settings()->primary_video
== PRIMARY_VIDEO_INTEL
) {
317 dev
= dev_find_device(PCI_VID_ASPEED
, PCI_DID_ASPEED_AST2050_VGA
, NULL
);
318 dev
->on_mainboard
= false;
319 dev
->enabled
= false;
320 dev
->ops
->init
= NULL
;
324 BOOT_STATE_INIT_ENTRY(BS_DEV_RESOURCES
, BS_ON_ENTRY
, mainboard_configure_internal_gfx
, NULL
)