From 6b95507ec5b087658178a325bdc68570bc48bb20 Mon Sep 17 00:00:00 2001 From: Bill XIE Date: Fri, 8 May 2020 16:40:48 +0800 Subject: [PATCH] mainboard/lenovo/x230: Add ThinkPad x230s as a variant The code is based on autoport and that for X230. Major differences are: - Only one DDR3 slot - HM77 PCH - M.2 socket instead of mini pci-e - no docking - no tpm Tested: - CPU i5-3337U - Slotted DIMM 8GiB - Camera - pci-e and usb2 on M.2 slot with A key for wlan - sata and usb2 (no superspeed components) on M.2 slot with B key for wwan - On board SDHCI connected to pci-e - USB3 ports - libgfxinit-based graphic init - NVRAM options for North and South bridges - Sound - Thinkpad EC - S3 - Linux 4.9 within Debian GNU/Linux stable, loaded from Seabios. Untested: - Touch screen, which is said to work under ubuntu but not debian. Change-Id: Ie537645d5ffaee799e79af2f821f80c3ebd2dfec Signed-off-by: Bill XIE Reviewed-on: https://review.coreboot.org/c/coreboot/+/41168 Tested-by: build bot (Jenkins) Reviewed-by: Paul Menzel Reviewed-by: Patrick Georgi --- 3rdparty/libgfxinit | 2 +- Documentation/mainboard/index.md | 1 + .../mainboard/lenovo/Ivy_Bridge_series.md | 2 +- Documentation/mainboard/lenovo/x230s.md | 15 ++ .../mainboard/lenovo/x230s_bc_removed.jpg | Bin 0 -> 42564 bytes src/mainboard/lenovo/x230/Kconfig | 23 +- src/mainboard/lenovo/x230/Kconfig.name | 3 + src/mainboard/lenovo/x230/Makefile.inc | 12 +- src/mainboard/lenovo/x230/board_info.txt | 1 + src/mainboard/lenovo/x230/devicetree.cb | 8 +- src/mainboard/lenovo/x230/hda_verb.c | 83 +------ .../lenovo/x230/variants/x230/board_info.txt | 7 + .../lenovo/x230/{ => variants/x230}/data.vbt | Bin .../x230/{ => variants/x230}/early_init.c | 0 .../{ => variants/x230}/gma-mainboard.ads | 0 .../lenovo/x230/{ => variants/x230}/gpio.c | 0 .../lenovo/x230/variants/x230/hda_verb.c | 82 +++++++ .../lenovo/x230/variants/x230/overridetree.cb | 15 ++ .../lenovo/x230/variants/x230s/board_info.txt | 7 + .../lenovo/x230/variants/x230s/data.vbt | Bin 0 -> 4280 bytes .../lenovo/x230/variants/x230s/early_init.c | 49 ++++ .../x230/variants/x230s/gma-mainboard.ads | 22 ++ .../lenovo/x230/variants/x230s/gpio.c | 212 ++++++++++++++++++ .../lenovo/x230/variants/x230s/hda_verb.c | 33 +++ .../x230/variants/x230s/overridetree.cb | 36 +++ 25 files changed, 512 insertions(+), 101 deletions(-) create mode 100644 Documentation/mainboard/lenovo/x230s.md create mode 100644 Documentation/mainboard/lenovo/x230s_bc_removed.jpg create mode 100644 src/mainboard/lenovo/x230/variants/x230/board_info.txt rename src/mainboard/lenovo/x230/{ => variants/x230}/data.vbt (100%) rename src/mainboard/lenovo/x230/{ => variants/x230}/early_init.c (100%) rename src/mainboard/lenovo/x230/{ => variants/x230}/gma-mainboard.ads (100%) rename src/mainboard/lenovo/x230/{ => variants/x230}/gpio.c (100%) create mode 100644 src/mainboard/lenovo/x230/variants/x230/hda_verb.c create mode 100644 src/mainboard/lenovo/x230/variants/x230/overridetree.cb create mode 100644 src/mainboard/lenovo/x230/variants/x230s/board_info.txt create mode 100644 src/mainboard/lenovo/x230/variants/x230s/data.vbt create mode 100644 src/mainboard/lenovo/x230/variants/x230s/early_init.c create mode 100644 src/mainboard/lenovo/x230/variants/x230s/gma-mainboard.ads create mode 100644 src/mainboard/lenovo/x230/variants/x230s/gpio.c create mode 100644 src/mainboard/lenovo/x230/variants/x230s/hda_verb.c create mode 100644 src/mainboard/lenovo/x230/variants/x230s/overridetree.cb diff --git a/3rdparty/libgfxinit b/3rdparty/libgfxinit index 2e87c0d40a..cdbfce2757 160000 --- a/3rdparty/libgfxinit +++ b/3rdparty/libgfxinit @@ -1 +1 @@ -Subproject commit 2e87c0d40a387c5b1f1afd3ce61ecdc7dad0e3e8 +Subproject commit cdbfce275777f2fd142e3a3c73469807a4c40207 diff --git a/Documentation/mainboard/index.md b/Documentation/mainboard/index.md index e80ff0b512..33c60a4c97 100644 --- a/Documentation/mainboard/index.md +++ b/Documentation/mainboard/index.md @@ -98,6 +98,7 @@ The boards in this section are not real mainboards, but emulators. - [W530](lenovo/w530.md) - [T430 / T530 / X230 / W530 common](lenovo/Ivy_Bridge_series.md) - [T431s](lenovo/t431s.md) +- [X230s](lenovo/x230s.md) - [Internal flashing](lenovo/ivb_internal_flashing.md) ### Haswell series diff --git a/Documentation/mainboard/lenovo/Ivy_Bridge_series.md b/Documentation/mainboard/lenovo/Ivy_Bridge_series.md index f4f0efff6c..5f151663c4 100644 --- a/Documentation/mainboard/lenovo/Ivy_Bridge_series.md +++ b/Documentation/mainboard/lenovo/Ivy_Bridge_series.md @@ -1,6 +1,6 @@ # Lenovo Ivy Bridge series -This information is valid for all supported models, except T430s and T431s. +This information is valid for all supported models, except T430s, [T431s](t431s.md) and [X230s](x230s.md). ## Flashing coreboot ```eval_rst diff --git a/Documentation/mainboard/lenovo/x230s.md b/Documentation/mainboard/lenovo/x230s.md new file mode 100644 index 0000000000..e42d75e974 --- /dev/null +++ b/Documentation/mainboard/lenovo/x230s.md @@ -0,0 +1,15 @@ +# Lenovo X230s + +## Disassembly Instructions + +You must remove the following parts to access the SPI flash chip: + +![x230s_bc_removed](x230s_bc_removed.jpg) + +* Base cover + +Its [Hardware Maintenance Manual](https://download.lenovo.com/ibmdl/pub/pc/pccbbs/mobiles_pdf/x230s_hmm_en_0c10860_01.pdf) could be used as a guidance of disassembly. + +The SPI flash chip (W25Q128.V in thr form of SOIC-8 for mine) is located at the circled place. Unlike [most Ivy Bridge ThinkPads](Ivy_Bridge_series.md), X230s has a single 16MiB SPI flash chip. + +The general [flashing tutorial](../../flash_tutorial/index.md) has more details. diff --git a/Documentation/mainboard/lenovo/x230s_bc_removed.jpg b/Documentation/mainboard/lenovo/x230s_bc_removed.jpg new file mode 100644 index 0000000000000000000000000000000000000000..1735e8100b82acbaeb56a8b4428b106408bd4b47 GIT binary patch literal 42564 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 - -const u32 cim_verb_data[] = { - /* --- Codec #0 --- */ - 0x10ec0269, /* Codec Vendor / Device ID: Realtek ALC269VC */ - 0x17aa21fa, /* Subsystem ID */ - 19, /* Number of 4 dword sets */ - AZALIA_SUBVENDOR(0, 0x17aa21fa), - - /* Ext. Microphone Connector: External,Right; MicIn,3.5mm; Black,JD; DA,Seq */ - AZALIA_PIN_CFG(0, 0x0a, 0x04a11020), - - /* Headphones Connector: External,Right; HP,3.5mm; Black,JD; DA,Seq */ - AZALIA_PIN_CFG(0, 0x0b, 0x0421101f), - - /* Not connected: N/A,N/A; Other,Unknown; Unknown,JD; DA,Seq */ - AZALIA_PIN_CFG(0, 0x0c, 0x40f000f0), - - /* Internal Speakers Fixed,Int; Speaker,Other Analog; Unknown,nJD; DA,Seq */ - AZALIA_PIN_CFG(0, 0x0d, 0x90170110), - - /* Not connected */ - AZALIA_PIN_CFG(0, 0x0f, 0x40f000f0), - - /* Internal Microphone: Fixed,Int,Top; Mic In,ATIPI; Unknown,nJD; DA,Seq */ - AZALIA_PIN_CFG(0, 0x11, 0xd5a30140), - AZALIA_PIN_CFG(0, 0x12, 0x90a60140), - AZALIA_PIN_CFG(0, 0x14, 0x90170110), - AZALIA_PIN_CFG(0, 0x15, 0x03211020), - AZALIA_PIN_CFG(0, 0x18, 0x03a11830), - AZALIA_PIN_CFG(0, 0x19, 0x411111f0), - AZALIA_PIN_CFG(0, 0x1a, 0x411111f0), - AZALIA_PIN_CFG(0, 0x1d, 0x40138205), - AZALIA_PIN_CFG(0, 0x1e, 0x411111f0), - - /* Misc entries */ - 0x01970804, - 0x01870803, - 0x01470740, - 0x00970640, - - 0x00370680, - 0x00270680, - 0x01470c02, - 0x01570c02, - - /* ALC coefficients. */ - /* 08 */ - 0x02050008, - 0x02040700, - /* 18 */ - 0x02050018, - 0x02045184, - /* 1c */ - 0x0205001c, - 0x02042800, - - 0x01870724, /* Enable Vrefout for mic */ - 0x00170500, /* Set power state to D0 */ - - /* --- Codec #3 --- */ - 0x80862806, /* Codec Vendor / Device ID: Intel PantherPoint HDMI */ - 0x80860101, /* Subsystem ID */ - 4, /* Number of 4 dword sets */ - AZALIA_SUBVENDOR(3, 0x80860101), - AZALIA_PIN_CFG(3, 0x05, 0x18560010), - AZALIA_PIN_CFG(3, 0x06, 0x18560020), - AZALIA_PIN_CFG(3, 0x07, 0x18560030), -}; - -const u32 pc_beep_verbs[] = { - 0x02177a00, /* Digital PCBEEP Gain: 0h=-9db, 1h=-6db ... 4h=+3db, 5h=+6db */ -}; - -AZALIA_ARRAY_SIZES; +/* dummy */ diff --git a/src/mainboard/lenovo/x230/variants/x230/board_info.txt b/src/mainboard/lenovo/x230/variants/x230/board_info.txt new file mode 100644 index 0000000000..22281e6aa8 --- /dev/null +++ b/src/mainboard/lenovo/x230/variants/x230/board_info.txt @@ -0,0 +1,7 @@ +Category: laptop +Board name: ThinkPad X230 +ROM package: SOIC-8 +ROM protocol: SPI +ROM socketed: n +Flashrom support: n +Release year: 2012 diff --git a/src/mainboard/lenovo/x230/data.vbt b/src/mainboard/lenovo/x230/variants/x230/data.vbt similarity index 100% rename from src/mainboard/lenovo/x230/data.vbt rename to src/mainboard/lenovo/x230/variants/x230/data.vbt diff --git a/src/mainboard/lenovo/x230/early_init.c b/src/mainboard/lenovo/x230/variants/x230/early_init.c similarity index 100% rename from src/mainboard/lenovo/x230/early_init.c rename to src/mainboard/lenovo/x230/variants/x230/early_init.c diff --git a/src/mainboard/lenovo/x230/gma-mainboard.ads b/src/mainboard/lenovo/x230/variants/x230/gma-mainboard.ads similarity index 100% rename from src/mainboard/lenovo/x230/gma-mainboard.ads rename to src/mainboard/lenovo/x230/variants/x230/gma-mainboard.ads diff --git a/src/mainboard/lenovo/x230/gpio.c b/src/mainboard/lenovo/x230/variants/x230/gpio.c similarity index 100% rename from src/mainboard/lenovo/x230/gpio.c rename to src/mainboard/lenovo/x230/variants/x230/gpio.c diff --git a/src/mainboard/lenovo/x230/variants/x230/hda_verb.c b/src/mainboard/lenovo/x230/variants/x230/hda_verb.c new file mode 100644 index 0000000000..05fb3fd775 --- /dev/null +++ b/src/mainboard/lenovo/x230/variants/x230/hda_verb.c @@ -0,0 +1,82 @@ +/* SPDX-License-Identifier: GPL-2.0-only */ + +/* Bits 31:28 - Codec Address */ +/* Bits 27:20 - NID */ +/* Bits 19:8 - Verb ID */ +/* Bits 7:0 - Payload */ + +#include + +const u32 cim_verb_data[] = { + /* --- Codec #0 --- */ + 0x10ec0269, /* Codec Vendor / Device ID: Realtek ALC269VC */ + 0x17aa21fa, /* Subsystem ID */ + 19, /* Number of 4 dword sets */ + AZALIA_SUBVENDOR(0, 0x17aa21fa), + + /* Ext. Microphone Connector: External,Right; MicIn,3.5mm; Black,JD; DA,Seq */ + AZALIA_PIN_CFG(0, 0x0a, 0x04a11020), + + /* Headphones Connector: External,Right; HP,3.5mm; Black,JD; DA,Seq */ + AZALIA_PIN_CFG(0, 0x0b, 0x0421101f), + + /* Not connected: N/A,N/A; Other,Unknown; Unknown,JD; DA,Seq */ + AZALIA_PIN_CFG(0, 0x0c, 0x40f000f0), + + /* Internal Speakers Fixed,Int; Speaker,Other Analog; Unknown,nJD; DA,Seq */ + AZALIA_PIN_CFG(0, 0x0d, 0x90170110), + + /* Not connected */ + AZALIA_PIN_CFG(0, 0x0f, 0x40f000f0), + + /* Internal Microphone: Fixed,Int,Top; Mic In,ATIPI; Unknown,nJD; DA,Seq */ + AZALIA_PIN_CFG(0, 0x11, 0xd5a30140), + AZALIA_PIN_CFG(0, 0x12, 0x90a60140), + AZALIA_PIN_CFG(0, 0x14, 0x90170110), + AZALIA_PIN_CFG(0, 0x15, 0x03211020), + AZALIA_PIN_CFG(0, 0x18, 0x03a11830), + AZALIA_PIN_CFG(0, 0x19, 0x411111f0), + AZALIA_PIN_CFG(0, 0x1a, 0x411111f0), + AZALIA_PIN_CFG(0, 0x1d, 0x40138205), + AZALIA_PIN_CFG(0, 0x1e, 0x411111f0), + + /* Misc entries */ + 0x01970804, + 0x01870803, + 0x01470740, + 0x00970640, + + 0x00370680, + 0x00270680, + 0x01470c02, + 0x01570c02, + + /* ALC coefficients. */ + /* 08 */ + 0x02050008, + 0x02040700, + /* 18 */ + 0x02050018, + 0x02045184, + /* 1c */ + 0x0205001c, + 0x02042800, + + 0x01870724, /* Enable Vrefout for mic */ + 0x00170500, /* Set power state to D0 */ + + /* --- Codec #3 --- */ + 0x80862806, /* Codec Vendor / Device ID: Intel PantherPoint HDMI */ + 0x80860101, /* Subsystem ID */ + 4, /* Number of 4 dword sets */ + AZALIA_SUBVENDOR(3, 0x80860101), + AZALIA_PIN_CFG(3, 0x05, 0x18560010), + AZALIA_PIN_CFG(3, 0x06, 0x18560020), + AZALIA_PIN_CFG(3, 0x07, 0x18560030), +}; + +const u32 pc_beep_verbs[] = { + 0x02177a00, /* Digital PCBEEP Gain: 0h=-9db, 1h=-6db ... 4h=+3db, 5h=+6db */ +}; + +AZALIA_ARRAY_SIZES; diff --git a/src/mainboard/lenovo/x230/variants/x230/overridetree.cb b/src/mainboard/lenovo/x230/variants/x230/overridetree.cb new file mode 100644 index 0000000000..8f1a97d9dd --- /dev/null +++ b/src/mainboard/lenovo/x230/variants/x230/overridetree.cb @@ -0,0 +1,15 @@ +chip northbridge/intel/sandybridge + device domain 0x0 on + chip southbridge/intel/bd82x6x # Intel Series 6 Cougar Point PCH + register "docking_supported" = "1" + device pci 1c.2 on + smbios_slot_desc "7" "3" "ExpressCard Slot" "8" + end # PCIe Port #3 (expresscard) + device pci 1f.0 on # LPC bridge + chip ec/lenovo/h8 + register "eventa_enable" = "0x01" + end + end # LPC Controller + end + end +end diff --git a/src/mainboard/lenovo/x230/variants/x230s/board_info.txt b/src/mainboard/lenovo/x230/variants/x230s/board_info.txt new file mode 100644 index 0000000000..67b229455a --- /dev/null +++ b/src/mainboard/lenovo/x230/variants/x230s/board_info.txt @@ -0,0 +1,7 @@ +Category: laptop +Board name: ThinkPad X230s +ROM package: SOIC-8 +ROM protocol: SPI +ROM socketed: n +Flashrom support: n +Release year: 2013 diff --git a/src/mainboard/lenovo/x230/variants/x230s/data.vbt b/src/mainboard/lenovo/x230/variants/x230s/data.vbt new file mode 100644 index 0000000000000000000000000000000000000000..42b0394daa90ec9b812d11b61284b91e8f09ef73 GIT binary patch literal 4280 zcmdT`U2GIp6h3!mc4u~WW_LO*uq?E>AmA1$oi4RPt#P{ju?zj#{!kY}q=j9OSfG@j zKtg2HiX|Gdi3U-T?3;x6P-8-TQHc-Os3F7$G(;j29}GUo14d(z_1u}+rM6f$5+dH& zZ_c@A&i&@zbMHB`i~3`Iw6`->9q*4-b#%qz?JYC~27IJ_kN4><7Vqla*xcCHxG~n= z)wG7b2eV;JyLAm9MM#+%t$B1se8-;jP^71l_73hy9oRIoEwwdW9pC3_G|?2Njhi>8 zhtea1d(tV|+y7J}Wwfds+q-QjMLYLy-;^Gyr;KZ3&$_`KIE2>JE?-egsj;fsc&s{V z)I?FQtzWUKenp)sH@7r(btK~LjeYU1PTJK&o8!HWy}d0RvG#Q|(bLrykH!01npL@J z=kS4%ZCjt+LnBR`;3d&;zF=g9DK+Bh^sdzjB|+LPX$9@&>pEmvh#1~|)0 z0!D;*K;%v0`!fp7g6S1F9|4(@6hYz%=OKbh1V7)GC+}Dz%xC2AoPb zBv~NAx&3Z*DM0oz((|m(nLtqh2yk%#G>x1oV}|Ty5x~RjqDClTirvs*!gk0Tm?V@d zbkX6`9USa;6`a6*=On{al}RDyo`7g@xCO(8N^w{7l>~VKOlZa{E85LGL#_ZAr>R55vveQhz`Wl zh(W}2h`oqIh$D#C5$__rd*Nn|vjDddn?+n%;z-a0I9dtT3Drv^U73)yuBdr-9agZ)mV^tsbRDNEOA^|3-HS;Di6Q$69FHPh2Hh> zytfL1uL2(R`__DR0W6wc8wCwmeK7$f47~+Jz*PW|fS5q>hlz-f1dvciBLx$69AYwm zfq)=^VA{h8O9tl=B_N?E{v_sa+Jx+p!%W#z3XC5gAx{ayZMwo}3aykvkSp}=sFa-q zZ`gSvHr&wzjsNA7*=umwyj{hl!rF|{06=?dVq;=(M|!ApaC^GQ&YU(#%>}a&mKdS2 zI%mBhV<{Z(2E#OlOC3wasp|b#y2DG%0(FOYC*DfkArIJ* zMvY|bd_&F4P6L24Hv)6v54`3=hOo_J*92~#b1lR3kk`V&#`!3DYRbE+ZDNAp$8THs3IR%=qHMPL6I*h6jWVT<@qX&sd}F(Kcmu7RX?W6S(Tns z^>0=As!FAR9uCNh0(5mi-w=?u1n6r4{bWEs6``jvouEkL!PzAz{+3#v=63l?{U zSS5JW@jyu<3Lc6ir`Yla(NKDV594=tdL;1-Ivb-D_ z#tWO$?*ZOnFlLy;Sb6`|1k5&;%&vp7cQk+y%MB+F@NcES7#e0=#q=a(IpyTfeowe4 zhW)fVteILNjae?*C0!|wVS3s#2Cp3l(>q?o9Hy5ERt4 +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +const struct southbridge_usb_port mainboard_usb_ports[] = { + { 1, 3, 0 }, + { 1, 3, 1 }, + { 0, 1, 3 }, + { 1, 3, -1 }, + { 0, 1, 2 }, + { 0, 1, -1 }, + { 0, 1, -1 }, + { 0, 1, -1 }, + { 0, 1, -1 }, + { 0, 1, 5 }, + { 1, 1, -1 }, + { 0, 1, -1 }, + { 1, 3, -1 }, + { 1, 1, -1 }, +}; + +void bootblock_mainboard_early_init(void) +{ + pci_write_config16(PCI_DEV(0, 0x1f, 0), 0x82, 0x3f0f); + pci_write_config16(PCI_DEV(0, 0x1f, 0), 0x80, 0x0010); +} + +/* FIXME: Put proper SPD map here. */ +void mainboard_get_spd(spd_raw_data *spd, bool id_only) +{ + read_spd(&spd[0], 0x50, id_only); + read_spd(&spd[1], 0x52, id_only); + read_spd(&spd[2], 0x51, id_only); + read_spd(&spd[3], 0x53, id_only); +} diff --git a/src/mainboard/lenovo/x230/variants/x230s/gma-mainboard.ads b/src/mainboard/lenovo/x230/variants/x230s/gma-mainboard.ads new file mode 100644 index 0000000000..fb75293b81 --- /dev/null +++ b/src/mainboard/lenovo/x230/variants/x230s/gma-mainboard.ads @@ -0,0 +1,22 @@ +-- SPDX-License-Identifier: GPL-2.0-or-later + +with HW.GFX.GMA; +with HW.GFX.GMA.Display_Probing; + +use HW.GFX.GMA; +use HW.GFX.GMA.Display_Probing; + +private package GMA.Mainboard is + + ports : constant Port_List := + (DP1, + DP2, + DP3, + HDMI1, + HDMI2, + HDMI3, + Analog, + EDP, + others => Disabled); + +end GMA.Mainboard; diff --git a/src/mainboard/lenovo/x230/variants/x230s/gpio.c b/src/mainboard/lenovo/x230/variants/x230s/gpio.c new file mode 100644 index 0000000000..a216c6bab0 --- /dev/null +++ b/src/mainboard/lenovo/x230/variants/x230s/gpio.c @@ -0,0 +1,212 @@ +/* SPDX-License-Identifier: GPL-2.0-only */ + +#include + +static const struct pch_gpio_set1 pch_gpio_set1_mode = { + .gpio0 = GPIO_MODE_GPIO, + .gpio1 = GPIO_MODE_GPIO, + .gpio2 = GPIO_MODE_GPIO, + .gpio3 = GPIO_MODE_GPIO, + .gpio4 = GPIO_MODE_GPIO, + .gpio5 = GPIO_MODE_GPIO, + .gpio6 = GPIO_MODE_GPIO, + .gpio7 = GPIO_MODE_GPIO, + .gpio8 = GPIO_MODE_GPIO, + .gpio9 = GPIO_MODE_NATIVE, + .gpio10 = GPIO_MODE_GPIO, + .gpio11 = GPIO_MODE_NATIVE, + .gpio12 = GPIO_MODE_NATIVE, + .gpio13 = GPIO_MODE_GPIO, + .gpio14 = GPIO_MODE_NATIVE, + .gpio15 = GPIO_MODE_GPIO, + .gpio16 = GPIO_MODE_GPIO, + .gpio17 = GPIO_MODE_GPIO, + .gpio18 = GPIO_MODE_NATIVE, + .gpio19 = GPIO_MODE_GPIO, + .gpio20 = GPIO_MODE_NATIVE, + .gpio21 = GPIO_MODE_GPIO, + .gpio22 = GPIO_MODE_GPIO, + .gpio23 = GPIO_MODE_NATIVE, + .gpio24 = GPIO_MODE_GPIO, + .gpio25 = GPIO_MODE_NATIVE, + .gpio26 = GPIO_MODE_GPIO, + .gpio27 = GPIO_MODE_GPIO, + .gpio28 = GPIO_MODE_GPIO, + .gpio29 = GPIO_MODE_GPIO, + .gpio30 = GPIO_MODE_NATIVE, + .gpio31 = GPIO_MODE_NATIVE, +}; + +static const struct pch_gpio_set1 pch_gpio_set1_direction = { + .gpio0 = GPIO_DIR_INPUT, + .gpio1 = GPIO_DIR_INPUT, + .gpio2 = GPIO_DIR_INPUT, + .gpio3 = GPIO_DIR_INPUT, + .gpio4 = GPIO_DIR_INPUT, + .gpio5 = GPIO_DIR_INPUT, + .gpio6 = GPIO_DIR_INPUT, + .gpio7 = GPIO_DIR_INPUT, + .gpio8 = GPIO_DIR_OUTPUT, + .gpio10 = GPIO_DIR_OUTPUT, + .gpio13 = GPIO_DIR_INPUT, + .gpio15 = GPIO_DIR_OUTPUT, + .gpio16 = GPIO_DIR_INPUT, + .gpio17 = GPIO_DIR_INPUT, + .gpio19 = GPIO_DIR_INPUT, + .gpio21 = GPIO_DIR_INPUT, + .gpio22 = GPIO_DIR_OUTPUT, + .gpio24 = GPIO_DIR_OUTPUT, + .gpio26 = GPIO_DIR_INPUT, + .gpio27 = GPIO_DIR_INPUT, + .gpio28 = GPIO_DIR_OUTPUT, + .gpio29 = GPIO_DIR_OUTPUT, +}; + +static const struct pch_gpio_set1 pch_gpio_set1_level = { + .gpio8 = GPIO_LEVEL_LOW, + .gpio10 = GPIO_LEVEL_HIGH, + .gpio15 = GPIO_LEVEL_LOW, + .gpio22 = GPIO_LEVEL_HIGH, + .gpio24 = GPIO_LEVEL_LOW, + .gpio28 = GPIO_LEVEL_LOW, + .gpio29 = GPIO_LEVEL_HIGH, +}; + +static const struct pch_gpio_set1 pch_gpio_set1_reset = { + .gpio24 = GPIO_RESET_RSMRST, +}; + +static const struct pch_gpio_set1 pch_gpio_set1_invert = { + .gpio1 = GPIO_INVERT, + .gpio6 = GPIO_INVERT, + .gpio13 = GPIO_INVERT, +}; + +static const struct pch_gpio_set1 pch_gpio_set1_blink = { +}; + +static const struct pch_gpio_set2 pch_gpio_set2_mode = { + .gpio32 = GPIO_MODE_NATIVE, + .gpio33 = GPIO_MODE_GPIO, + .gpio34 = GPIO_MODE_GPIO, + .gpio35 = GPIO_MODE_GPIO, + .gpio36 = GPIO_MODE_GPIO, + .gpio37 = GPIO_MODE_GPIO, + .gpio38 = GPIO_MODE_GPIO, + .gpio39 = GPIO_MODE_GPIO, + .gpio40 = GPIO_MODE_NATIVE, + .gpio41 = GPIO_MODE_NATIVE, + .gpio42 = GPIO_MODE_NATIVE, + .gpio43 = GPIO_MODE_GPIO, + .gpio44 = GPIO_MODE_GPIO, + .gpio45 = GPIO_MODE_GPIO, + .gpio46 = GPIO_MODE_NATIVE, + .gpio47 = GPIO_MODE_GPIO, + .gpio48 = GPIO_MODE_GPIO, + .gpio49 = GPIO_MODE_GPIO, + .gpio50 = GPIO_MODE_GPIO, + .gpio51 = GPIO_MODE_GPIO, + .gpio52 = GPIO_MODE_GPIO, + .gpio53 = GPIO_MODE_GPIO, + .gpio54 = GPIO_MODE_GPIO, + .gpio55 = GPIO_MODE_GPIO, + .gpio56 = GPIO_MODE_GPIO, + .gpio57 = GPIO_MODE_GPIO, + .gpio58 = GPIO_MODE_NATIVE, + .gpio59 = GPIO_MODE_NATIVE, + .gpio60 = GPIO_MODE_NATIVE, + .gpio61 = GPIO_MODE_NATIVE, + .gpio62 = GPIO_MODE_NATIVE, + .gpio63 = GPIO_MODE_NATIVE, +}; + +static const struct pch_gpio_set2 pch_gpio_set2_direction = { + .gpio33 = GPIO_DIR_OUTPUT, + .gpio34 = GPIO_DIR_INPUT, + .gpio35 = GPIO_DIR_INPUT, + .gpio36 = GPIO_DIR_INPUT, + .gpio37 = GPIO_DIR_INPUT, + .gpio38 = GPIO_DIR_INPUT, + .gpio39 = GPIO_DIR_INPUT, + .gpio43 = GPIO_DIR_OUTPUT, + .gpio44 = GPIO_DIR_INPUT, + .gpio45 = GPIO_DIR_INPUT, + .gpio47 = GPIO_DIR_INPUT, + .gpio48 = GPIO_DIR_INPUT, + .gpio49 = GPIO_DIR_INPUT, + .gpio50 = GPIO_DIR_INPUT, + .gpio51 = GPIO_DIR_OUTPUT, + .gpio52 = GPIO_DIR_OUTPUT, + .gpio53 = GPIO_DIR_OUTPUT, + .gpio54 = GPIO_DIR_INPUT, + .gpio55 = GPIO_DIR_OUTPUT, + .gpio56 = GPIO_DIR_INPUT, + .gpio57 = GPIO_DIR_INPUT, +}; + +static const struct pch_gpio_set2 pch_gpio_set2_level = { + .gpio33 = GPIO_LEVEL_HIGH, + .gpio43 = GPIO_LEVEL_HIGH, + .gpio51 = GPIO_LEVEL_HIGH, + .gpio52 = GPIO_LEVEL_HIGH, + .gpio53 = GPIO_LEVEL_HIGH, + .gpio55 = GPIO_LEVEL_HIGH, +}; + +static const struct pch_gpio_set2 pch_gpio_set2_reset = { +}; + +static const struct pch_gpio_set3 pch_gpio_set3_mode = { + .gpio64 = GPIO_MODE_GPIO, + .gpio65 = GPIO_MODE_GPIO, + .gpio66 = GPIO_MODE_GPIO, + .gpio67 = GPIO_MODE_GPIO, + .gpio68 = GPIO_MODE_GPIO, + .gpio69 = GPIO_MODE_GPIO, + .gpio70 = GPIO_MODE_GPIO, + .gpio71 = GPIO_MODE_GPIO, + .gpio72 = GPIO_MODE_NATIVE, + .gpio73 = GPIO_MODE_NATIVE, + .gpio74 = GPIO_MODE_NATIVE, + .gpio75 = GPIO_MODE_NATIVE, +}; + +static const struct pch_gpio_set3 pch_gpio_set3_direction = { + .gpio64 = GPIO_DIR_INPUT, + .gpio65 = GPIO_DIR_INPUT, + .gpio66 = GPIO_DIR_INPUT, + .gpio67 = GPIO_DIR_INPUT, + .gpio68 = GPIO_DIR_INPUT, + .gpio69 = GPIO_DIR_INPUT, + .gpio70 = GPIO_DIR_INPUT, + .gpio71 = GPIO_DIR_INPUT, +}; + +static const struct pch_gpio_set3 pch_gpio_set3_level = { +}; + +static const struct pch_gpio_set3 pch_gpio_set3_reset = { +}; + +const struct pch_gpio_map mainboard_gpio_map = { + .set1 = { + .mode = &pch_gpio_set1_mode, + .direction = &pch_gpio_set1_direction, + .level = &pch_gpio_set1_level, + .blink = &pch_gpio_set1_blink, + .invert = &pch_gpio_set1_invert, + .reset = &pch_gpio_set1_reset, + }, + .set2 = { + .mode = &pch_gpio_set2_mode, + .direction = &pch_gpio_set2_direction, + .level = &pch_gpio_set2_level, + .reset = &pch_gpio_set2_reset, + }, + .set3 = { + .mode = &pch_gpio_set3_mode, + .direction = &pch_gpio_set3_direction, + .level = &pch_gpio_set3_level, + .reset = &pch_gpio_set3_reset, + }, +}; diff --git a/src/mainboard/lenovo/x230/variants/x230s/hda_verb.c b/src/mainboard/lenovo/x230/variants/x230s/hda_verb.c new file mode 100644 index 0000000000..77919041e5 --- /dev/null +++ b/src/mainboard/lenovo/x230/variants/x230s/hda_verb.c @@ -0,0 +1,33 @@ +/* SPDX-License-Identifier: GPL-2.0-only */ + +#include + +const u32 cim_verb_data[] = { + 0x10ec0269, /* Codec Vendor / Device ID: Realtek */ + 0x17aa2209, /* Subsystem ID */ + 11, /* Number of 4 dword sets */ + AZALIA_SUBVENDOR(0, 0x17aa2209), + AZALIA_PIN_CFG(0, 0x12, 0x90a60140), + AZALIA_PIN_CFG(0, 0x14, 0x90170110), + AZALIA_PIN_CFG(0, 0x15, 0x03211020), + AZALIA_PIN_CFG(0, 0x17, 0x40008000), + AZALIA_PIN_CFG(0, 0x18, 0x03a11030), + AZALIA_PIN_CFG(0, 0x19, 0x411111f0), + AZALIA_PIN_CFG(0, 0x1a, 0x411111f0), + AZALIA_PIN_CFG(0, 0x1b, 0x411111f0), + AZALIA_PIN_CFG(0, 0x1d, 0x40f38205), + AZALIA_PIN_CFG(0, 0x1e, 0x411111f0), + + 0x80862806, /* Codec Vendor / Device ID: Intel */ + 0x80860101, /* Subsystem ID */ + 4, /* Number of 4 dword sets */ + AZALIA_SUBVENDOR(3, 0x80860101), + AZALIA_PIN_CFG(3, 0x05, 0x18560010), + AZALIA_PIN_CFG(3, 0x06, 0x58560020), + AZALIA_PIN_CFG(3, 0x07, 0x58560030), + +}; + +const u32 pc_beep_verbs[0] = {}; + +AZALIA_ARRAY_SIZES; diff --git a/src/mainboard/lenovo/x230/variants/x230s/overridetree.cb b/src/mainboard/lenovo/x230/variants/x230s/overridetree.cb new file mode 100644 index 0000000000..ed1dd3c3b4 --- /dev/null +++ b/src/mainboard/lenovo/x230/variants/x230s/overridetree.cb @@ -0,0 +1,36 @@ +chip northbridge/intel/sandybridge + # Enable DisplayPort Hotplug with 6ms pulse + register "gpu_dp_b_hotplug" = "4" + register "gpu_dp_c_hotplug" = "4" + register "gpu_dp_d_hotplug" = "4" + + # Enable Panel as eDP and configure power delays + register "gpu_panel_port_select" = "1" # eDP + register "gpu_panel_power_backlight_off_delay" = "1" # 0.1ms + register "gpu_panel_power_backlight_on_delay" = "1" # 0.1ms + register "gpu_panel_power_down_delay" = "500" # 50ms + register "gpu_panel_power_up_delay" = "2000" # 200ms + + device domain 0x0 on + subsystemid 0x17aa 0x2209 inherit + chip southbridge/intel/bd82x6x # Intel Series 6 Cougar Point PCH + register "c2_latency" = "0x0065" + # X230s does not support docking + register "docking_supported" = "0" + register "pcie_hotplug_map" = "{ 0, 0, 0, 0, 0, 0, 0, 0 }" + # Enable SATA ports 0 (HDD bay) & 1 (WWAN M.2 SATA) + register "sata_port_map" = "0x3" + + device pci 1f.0 on # LPC bridge + chip ec/lenovo/h8 # + register "config1" = "0x05" + register "config3" = "0xc4" + register "event5_enable" = "0x3c" + register "evente_enable" = "0x1d" + # X230s only has BT on wlan card + register "has_bdc_detection" = "0" + end + end # LPC Controller + end + end +end