Angel Pons | feedf23 | 2020-04-05 13:22:01 +0200 | [diff] [blame] | 1 | /* SPDX-License-Identifier: GPL-2.0-only */ |
Stefan Reinauer | 49428d8 | 2013-02-21 15:48:37 -0800 | [diff] [blame] | 2 | |
| 3 | #include <types.h> |
Stefan Reinauer | 49428d8 | 2013-02-21 15:48:37 -0800 | [diff] [blame] | 4 | #include <device/device.h> |
Stefan Reinauer | 49428d8 | 2013-02-21 15:48:37 -0800 | [diff] [blame] | 5 | #include <device/pci_ops.h> |
Kyösti Mälkki | 383c4e7 | 2023-01-05 18:05:11 +0200 | [diff] [blame] | 6 | #include <ec/ec.h> |
Stefan Reinauer | 49428d8 | 2013-02-21 15:48:37 -0800 | [diff] [blame] | 7 | #include <console/console.h> |
Julius Werner | cd49cce | 2019-03-05 16:53:33 -0800 | [diff] [blame] | 8 | #if CONFIG(VGA_ROM_RUN) |
Stefan Reinauer | 49428d8 | 2013-02-21 15:48:37 -0800 | [diff] [blame] | 9 | #include <x86emu/x86emu.h> |
| 10 | #endif |
Stefan Reinauer | 49428d8 | 2013-02-21 15:48:37 -0800 | [diff] [blame] | 11 | #include <arch/io.h> |
| 12 | #include <arch/interrupt.h> |
Stefan Reinauer | 49428d8 | 2013-02-21 15:48:37 -0800 | [diff] [blame] | 13 | #include "onboard.h" |
| 14 | #include "ec.h" |
| 15 | #include <southbridge/intel/bd82x6x/pch.h> |
Patrick Rudolph | e8e66f4 | 2016-02-06 17:42:42 +0100 | [diff] [blame] | 16 | #include <southbridge/intel/common/gpio.h> |
Stefan Reinauer | 49428d8 | 2013-02-21 15:48:37 -0800 | [diff] [blame] | 17 | #include <smbios.h> |
Stefan Reinauer | 49428d8 | 2013-02-21 15:48:37 -0800 | [diff] [blame] | 18 | #include <ec/google/chromeec/ec.h> |
| 19 | |
Julius Werner | cd49cce | 2019-03-05 16:53:33 -0800 | [diff] [blame] | 20 | #if CONFIG(VGA_ROM_RUN) |
Stefan Reinauer | fe8290d | 2013-04-23 15:00:02 -0700 | [diff] [blame] | 21 | static int int15_handler(void) |
Stefan Reinauer | 49428d8 | 2013-02-21 15:48:37 -0800 | [diff] [blame] | 22 | { |
Duncan Laurie | 1c05400 | 2013-05-10 13:50:09 -0700 | [diff] [blame] | 23 | int res = 0; |
Stefan Reinauer | 49428d8 | 2013-02-21 15:48:37 -0800 | [diff] [blame] | 24 | |
| 25 | printk(BIOS_DEBUG, "%s: INT15 function %04x!\n", |
Stefan Reinauer | fe8290d | 2013-04-23 15:00:02 -0700 | [diff] [blame] | 26 | __func__, X86_AX); |
Stefan Reinauer | 49428d8 | 2013-02-21 15:48:37 -0800 | [diff] [blame] | 27 | |
Elyes HAOUAS | 0ce41f1 | 2018-11-13 10:03:31 +0100 | [diff] [blame] | 28 | switch (X86_AX) { |
Stefan Reinauer | 49428d8 | 2013-02-21 15:48:37 -0800 | [diff] [blame] | 29 | case 0x5f34: |
| 30 | /* |
| 31 | * Set Panel Fitting Hook: |
| 32 | * bit 2 = Graphics Stretching |
| 33 | * bit 1 = Text Stretching |
| 34 | * bit 0 = Centering (do not set with bit1 or bit2) |
Elyes HAOUAS | 6dc9d03 | 2020-02-16 16:22:52 +0100 | [diff] [blame] | 35 | * 0 = video BIOS default |
Stefan Reinauer | 49428d8 | 2013-02-21 15:48:37 -0800 | [diff] [blame] | 36 | */ |
Stefan Reinauer | fe8290d | 2013-04-23 15:00:02 -0700 | [diff] [blame] | 37 | X86_AX = 0x005f; |
Elyes HAOUAS | 6dc9d03 | 2020-02-16 16:22:52 +0100 | [diff] [blame] | 38 | X86_CL = 0x00; /* Use video BIOS default */ |
Duncan Laurie | 1c05400 | 2013-05-10 13:50:09 -0700 | [diff] [blame] | 39 | res = 1; |
Stefan Reinauer | 49428d8 | 2013-02-21 15:48:37 -0800 | [diff] [blame] | 40 | break; |
| 41 | case 0x5f35: |
| 42 | /* |
| 43 | * Boot Display Device Hook: |
| 44 | * bit 0 = CRT |
| 45 | * bit 1 = TV (eDP) |
| 46 | * bit 2 = EFP |
| 47 | * bit 3 = LFP |
| 48 | * bit 4 = CRT2 |
| 49 | * bit 5 = TV2 (eDP) |
| 50 | * bit 6 = EFP2 |
| 51 | * bit 7 = LFP2 |
| 52 | */ |
Stefan Reinauer | fe8290d | 2013-04-23 15:00:02 -0700 | [diff] [blame] | 53 | X86_AX = 0x005f; |
Elyes HAOUAS | 6dc9d03 | 2020-02-16 16:22:52 +0100 | [diff] [blame] | 54 | X86_CX = 0x0000; /* Use video BIOS default */ |
Duncan Laurie | 1c05400 | 2013-05-10 13:50:09 -0700 | [diff] [blame] | 55 | res = 1; |
Stefan Reinauer | 49428d8 | 2013-02-21 15:48:37 -0800 | [diff] [blame] | 56 | break; |
| 57 | case 0x5f51: |
| 58 | /* |
| 59 | * Hook to select active LFP configuration: |
| 60 | * 00h = No LVDS, VBIOS does not enable LVDS |
| 61 | * 01h = Int-LVDS, LFP driven by integrated LVDS decoder |
| 62 | * 02h = SVDO-LVDS, LFP driven by SVDO decoder |
| 63 | * 03h = eDP, LFP Driven by Int-DisplayPort encoder |
| 64 | */ |
Stefan Reinauer | fe8290d | 2013-04-23 15:00:02 -0700 | [diff] [blame] | 65 | X86_AX = 0x005f; |
| 66 | X86_CX = 0x0003; /* eDP */ |
Duncan Laurie | 1c05400 | 2013-05-10 13:50:09 -0700 | [diff] [blame] | 67 | res = 1; |
Stefan Reinauer | 49428d8 | 2013-02-21 15:48:37 -0800 | [diff] [blame] | 68 | break; |
| 69 | case 0x5f70: |
Stefan Reinauer | fe8290d | 2013-04-23 15:00:02 -0700 | [diff] [blame] | 70 | switch (X86_CH) { |
Stefan Reinauer | 49428d8 | 2013-02-21 15:48:37 -0800 | [diff] [blame] | 71 | case 0: |
| 72 | /* Get Mux */ |
Stefan Reinauer | fe8290d | 2013-04-23 15:00:02 -0700 | [diff] [blame] | 73 | X86_AX = 0x005f; |
| 74 | X86_CX = 0x0000; |
Duncan Laurie | 1c05400 | 2013-05-10 13:50:09 -0700 | [diff] [blame] | 75 | res = 1; |
Stefan Reinauer | 49428d8 | 2013-02-21 15:48:37 -0800 | [diff] [blame] | 76 | break; |
| 77 | case 1: |
| 78 | /* Set Mux */ |
Stefan Reinauer | fe8290d | 2013-04-23 15:00:02 -0700 | [diff] [blame] | 79 | X86_AX = 0x005f; |
| 80 | X86_CX = 0x0000; |
Duncan Laurie | 1c05400 | 2013-05-10 13:50:09 -0700 | [diff] [blame] | 81 | res = 1; |
Stefan Reinauer | 49428d8 | 2013-02-21 15:48:37 -0800 | [diff] [blame] | 82 | break; |
| 83 | case 2: |
| 84 | /* Get SG/Non-SG mode */ |
Stefan Reinauer | fe8290d | 2013-04-23 15:00:02 -0700 | [diff] [blame] | 85 | X86_AX = 0x005f; |
| 86 | X86_CX = 0x0000; |
Duncan Laurie | 1c05400 | 2013-05-10 13:50:09 -0700 | [diff] [blame] | 87 | res = 1; |
Stefan Reinauer | 49428d8 | 2013-02-21 15:48:37 -0800 | [diff] [blame] | 88 | break; |
| 89 | default: |
| 90 | /* Interrupt was not handled */ |
| 91 | printk(BIOS_DEBUG, "Unknown INT15 5f70 function: 0x%02x\n", |
Stefan Reinauer | fe8290d | 2013-04-23 15:00:02 -0700 | [diff] [blame] | 92 | X86_CH); |
Duncan Laurie | 1c05400 | 2013-05-10 13:50:09 -0700 | [diff] [blame] | 93 | break; |
Stefan Reinauer | 49428d8 | 2013-02-21 15:48:37 -0800 | [diff] [blame] | 94 | } |
| 95 | break; |
| 96 | case 0x5fac: |
Duncan Laurie | 1c05400 | 2013-05-10 13:50:09 -0700 | [diff] [blame] | 97 | res = 1; |
Stefan Reinauer | 49428d8 | 2013-02-21 15:48:37 -0800 | [diff] [blame] | 98 | break; |
| 99 | default: |
Stefan Reinauer | fe8290d | 2013-04-23 15:00:02 -0700 | [diff] [blame] | 100 | printk(BIOS_DEBUG, "Unknown INT15 function %04x!\n", X86_AX); |
Stefan Reinauer | 49428d8 | 2013-02-21 15:48:37 -0800 | [diff] [blame] | 101 | break; |
| 102 | } |
| 103 | return res; |
| 104 | } |
| 105 | #endif |
| 106 | |
Elyes HAOUAS | d129d43 | 2018-05-04 20:23:33 +0200 | [diff] [blame] | 107 | static void mainboard_init(struct device *dev) |
Stefan Reinauer | 49428d8 | 2013-02-21 15:48:37 -0800 | [diff] [blame] | 108 | { |
Karthikeyan Ramasubramanian | c80ff84 | 2018-09-17 16:19:34 -0600 | [diff] [blame] | 109 | uint32_t board_version = 0; |
| 110 | |
Stefan Reinauer | 49428d8 | 2013-02-21 15:48:37 -0800 | [diff] [blame] | 111 | /* Initialize the Embedded Controller */ |
Kyösti Mälkki | 383c4e7 | 2023-01-05 18:05:11 +0200 | [diff] [blame] | 112 | mainboard_ec_init(); |
Stefan Reinauer | 49428d8 | 2013-02-21 15:48:37 -0800 | [diff] [blame] | 113 | |
Karthikeyan Ramasubramanian | c80ff84 | 2018-09-17 16:19:34 -0600 | [diff] [blame] | 114 | google_chromeec_get_board_version(&board_version); |
| 115 | if (board_version == 0) { |
Stefan Reinauer | 49428d8 | 2013-02-21 15:48:37 -0800 | [diff] [blame] | 116 | /* If running on proto1 - enable reversion of gpio11. */ |
| 117 | u32 gpio_inv; |
| 118 | u16 gpio_base = pci_read_config16 |
Kyösti Mälkki | c70eed1 | 2018-05-22 02:18:00 +0300 | [diff] [blame] | 119 | (pcidev_on_root(0x1f, 0), GPIO_BASE) & |
Stefan Reinauer | 49428d8 | 2013-02-21 15:48:37 -0800 | [diff] [blame] | 120 | 0xfffc; |
| 121 | u16 gpio_inv_addr = gpio_base + GPI_INV; |
| 122 | gpio_inv = inl(gpio_inv_addr); |
| 123 | outl(gpio_inv | (1 << 11), gpio_inv_addr); |
| 124 | } |
| 125 | } |
| 126 | |
Elyes HAOUAS | d129d43 | 2018-05-04 20:23:33 +0200 | [diff] [blame] | 127 | static int link_onboard_smbios_data(struct device *dev, int *handle, |
Stefan Reinauer | 49428d8 | 2013-02-21 15:48:37 -0800 | [diff] [blame] | 128 | unsigned long *current) |
| 129 | { |
| 130 | int len = 0; |
| 131 | |
Duncan Laurie | 21a7870 | 2013-05-23 14:17:05 -0700 | [diff] [blame] | 132 | len += smbios_write_type41( |
| 133 | current, handle, |
Kyösti Mälkki | b9cd5ec | 2015-04-24 16:05:58 +0300 | [diff] [blame] | 134 | BOARD_LIGHTSENSOR_NAME, /* name */ |
| 135 | BOARD_LIGHTSENSOR_IRQ, /* instance */ |
Duncan Laurie | 21a7870 | 2013-05-23 14:17:05 -0700 | [diff] [blame] | 136 | 0, /* segment */ |
Kyösti Mälkki | b9cd5ec | 2015-04-24 16:05:58 +0300 | [diff] [blame] | 137 | BOARD_LIGHTSENSOR_I2C_ADDR, /* bus */ |
Duncan Laurie | 21a7870 | 2013-05-23 14:17:05 -0700 | [diff] [blame] | 138 | 0, /* device */ |
Christian Walter | e6afab1 | 2019-05-21 17:22:49 +0200 | [diff] [blame] | 139 | 0, /* function */ |
| 140 | SMBIOS_DEVICE_TYPE_OTHER); /* device type */ |
Stefan Reinauer | 49428d8 | 2013-02-21 15:48:37 -0800 | [diff] [blame] | 141 | |
Duncan Laurie | 21a7870 | 2013-05-23 14:17:05 -0700 | [diff] [blame] | 142 | len += smbios_write_type41( |
| 143 | current, handle, |
Kyösti Mälkki | b9cd5ec | 2015-04-24 16:05:58 +0300 | [diff] [blame] | 144 | BOARD_TRACKPAD_NAME, /* name */ |
| 145 | BOARD_TRACKPAD_IRQ, /* instance */ |
Duncan Laurie | 21a7870 | 2013-05-23 14:17:05 -0700 | [diff] [blame] | 146 | 0, /* segment */ |
Kyösti Mälkki | b9cd5ec | 2015-04-24 16:05:58 +0300 | [diff] [blame] | 147 | BOARD_TRACKPAD_I2C_ADDR, /* bus */ |
Duncan Laurie | 21a7870 | 2013-05-23 14:17:05 -0700 | [diff] [blame] | 148 | 0, /* device */ |
Christian Walter | e6afab1 | 2019-05-21 17:22:49 +0200 | [diff] [blame] | 149 | 0, /* function */ |
| 150 | SMBIOS_DEVICE_TYPE_OTHER); /* device type */ |
Stefan Reinauer | 49428d8 | 2013-02-21 15:48:37 -0800 | [diff] [blame] | 151 | |
Duncan Laurie | 21a7870 | 2013-05-23 14:17:05 -0700 | [diff] [blame] | 152 | len += smbios_write_type41( |
| 153 | current, handle, |
Kyösti Mälkki | b9cd5ec | 2015-04-24 16:05:58 +0300 | [diff] [blame] | 154 | BOARD_TOUCHSCREEN_NAME, /* name */ |
| 155 | BOARD_TOUCHSCREEN_IRQ, /* instance */ |
Duncan Laurie | 21a7870 | 2013-05-23 14:17:05 -0700 | [diff] [blame] | 156 | 0, /* segment */ |
Kyösti Mälkki | b9cd5ec | 2015-04-24 16:05:58 +0300 | [diff] [blame] | 157 | BOARD_TOUCHSCREEN_I2C_ADDR, /* bus */ |
Duncan Laurie | 21a7870 | 2013-05-23 14:17:05 -0700 | [diff] [blame] | 158 | 0, /* device */ |
Christian Walter | e6afab1 | 2019-05-21 17:22:49 +0200 | [diff] [blame] | 159 | 0, /* function */ |
| 160 | SMBIOS_DEVICE_TYPE_OTHER); /* device type */ |
Stefan Reinauer | 49428d8 | 2013-02-21 15:48:37 -0800 | [diff] [blame] | 161 | |
| 162 | return len; |
| 163 | } |
| 164 | |
Elyes HAOUAS | d129d43 | 2018-05-04 20:23:33 +0200 | [diff] [blame] | 165 | static void mainboard_enable(struct device *dev) |
Stefan Reinauer | 49428d8 | 2013-02-21 15:48:37 -0800 | [diff] [blame] | 166 | { |
| 167 | dev->ops->init = mainboard_init; |
| 168 | dev->ops->get_smbios_data = link_onboard_smbios_data; |
Julius Werner | cd49cce | 2019-03-05 16:53:33 -0800 | [diff] [blame] | 169 | #if CONFIG(VGA_ROM_RUN) |
Stefan Reinauer | 49428d8 | 2013-02-21 15:48:37 -0800 | [diff] [blame] | 170 | /* Install custom int15 handler for VGA OPROM */ |
Stefan Reinauer | fe8290d | 2013-04-23 15:00:02 -0700 | [diff] [blame] | 171 | mainboard_interrupt_handlers(0x15, &int15_handler); |
Stefan Reinauer | 49428d8 | 2013-02-21 15:48:37 -0800 | [diff] [blame] | 172 | #endif |
Stefan Reinauer | 49428d8 | 2013-02-21 15:48:37 -0800 | [diff] [blame] | 173 | } |
| 174 | |
| 175 | struct chip_operations mainboard_ops = { |
Stefan Reinauer | 49428d8 | 2013-02-21 15:48:37 -0800 | [diff] [blame] | 176 | .enable_dev = mainboard_enable, |
| 177 | }; |