Vladimir Serbinenko (phcoder@gmail.com) just uploaded a new patch set to gerrit, which you can find at http://review.coreboot.org/4774
-gerrit
commit bfa57c7582a516ff0d7fa03949acefb92dc70bd2 Author: Vladimir Serbinenko phcoder@gmail.com Date: Thu Jan 23 09:06:08 2014 +0100
lenovo: Handle EEPROM/RFID chip.
EEPROM/RFID chip present in thinkpad should be locked in a way to avoid any potential RFID access.
Read serial number, UUID and P/N from EEPROM.
This info is stored on AT24RF08 chip acessible through SMBUS.
Change-Id: Ia3e766d90a094f63c8c854cd37e165221ccd8acd Signed-off-by: Vladimir Serbinenko phcoder@gmail.com --- src/drivers/i2c/Makefile.inc | 1 + src/drivers/i2c/lenovo/Makefile.inc | 1 + src/drivers/i2c/lenovo/eeprom.c | 167 ++++++++++++++++++++++++++++++++ src/mainboard/lenovo/t60/devicetree.cb | 25 +++++ src/mainboard/lenovo/x201/devicetree.cb | 25 +++++ src/mainboard/lenovo/x230/devicetree.cb | 28 +++++- src/mainboard/lenovo/x60/devicetree.cb | 26 ++++- 7 files changed, 271 insertions(+), 2 deletions(-)
diff --git a/src/drivers/i2c/Makefile.inc b/src/drivers/i2c/Makefile.inc index ef7ac4b..2cd1db0 100644 --- a/src/drivers/i2c/Makefile.inc +++ b/src/drivers/i2c/Makefile.inc @@ -7,3 +7,4 @@ subdirs-y += lm63 subdirs-y += rtd2132 subdirs-y += w83795 subdirs-y += w83793 +subdirs-y += lenovo diff --git a/src/drivers/i2c/lenovo/Makefile.inc b/src/drivers/i2c/lenovo/Makefile.inc new file mode 100644 index 0000000..87f0dbf --- /dev/null +++ b/src/drivers/i2c/lenovo/Makefile.inc @@ -0,0 +1 @@ +ramstage-$(CONFIG_VENDOR_LENOVO) += eeprom.c diff --git a/src/drivers/i2c/lenovo/eeprom.c b/src/drivers/i2c/lenovo/eeprom.c new file mode 100644 index 0000000..1a8512c --- /dev/null +++ b/src/drivers/i2c/lenovo/eeprom.c @@ -0,0 +1,167 @@ +/* + * This file is part of the coreboot project. + * + * Copyright (C) 2013 Vladimir Serbinenko + * + * 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. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA + */ + +#include <types.h> +#include <string.h> +#include <arch/io.h> +#include <device/device.h> +#include <device/smbus.h> +#include <smbios.h> +#include <console/console.h> + +static void smbus_read_string(device_t dev, u8 start, u8 len, char *result) +{ + int i; + + for (i = 0; i < len; i++) { + int t = smbus_read_byte(dev, start + i); + if (t < 0x20 || t > 0x7f) { + memcpy(result, "*INVALID*", sizeof ("*INVALID*")); + return; + } + result[i] = t; + } +} + +const char *smbios_mainboard_serial_number(void) +{ + static char result[12]; + static int already_read; + device_t dev; + + if (already_read) + return result; + + memset(result, 0, sizeof (result)); + + dev = dev_find_slot_on_smbus(1, 0x54); + if (dev == 0) { + printk(BIOS_WARNING, "eeprom not found\n"); + already_read = 1; + return result; + } + + smbus_read_string(dev, 0x2e, 7, result); + already_read = 1; + return result; +} + +const char *smbios_mainboard_product_name(void) +{ + static char result[12]; + static int already_read; + device_t dev; + + if (already_read) + return result; + memset (result, 0, sizeof (result)); + + dev = dev_find_slot_on_smbus(1, 0x54); + if (dev == 0) { + printk(BIOS_WARNING, "eeprom not found\n"); + already_read = 1; + return result; + } + + smbus_read_string(dev, 0x27, 7, result); + + already_read = 1; + return result; +} + +void smbios_mainboard_set_uuid(u8 *uuid) +{ + static char result[16]; + unsigned i; + static int already_read; + device_t dev; + const int remap[16] = { + /* UUID byteswap. */ + 3, 2, 1, 0, 5, 4, 7, 6, 8, 9, 10, 11, 12, 13, 14, 15 + }; + + + if (already_read) { + memcpy (uuid, result, 16); + return; + } + + memset (result, 0, sizeof (result)); + + dev = dev_find_slot_on_smbus(1, 0x56); + if (dev == 0) { + printk(BIOS_WARNING, "eeprom not found\n"); + already_read = 1; + memset (uuid, 0, 16); + return; + } + + for (i = 0; i < 16; i++) { + int t = smbus_read_byte(dev, 0x12 + i); + if (t < 0) { + memset (result, 0, sizeof (result)); + break; + } + result[remap[i]] = t; + } + + already_read = 1; + + memcpy (uuid, result, 16); +} + +static void lenovo_eeprom_init(device_t dev) +{ + int i; + + if (!dev->enabled) + return; + + /* Ensure that EEPROM/RFID chip is not accessible through RFID. + Need to do it only on 5c. */ + if (dev->path.type != DEVICE_PATH_I2C || dev->path.i2c.device != 0x5c) + return; + + printk (BIOS_DEBUG, "Locking EEPROM RFID\n"); + + for (i = 0; i < 8; i++) + smbus_write_byte(dev, i, 0x0f); + printk (BIOS_DEBUG, "init EEPROM done\n"); +} + +static void lenovo_eeprom_noop(device_t dummy) +{ +} + +static struct device_operations lenovo_eeprom_operations = { + .read_resources = lenovo_eeprom_noop, + .set_resources = lenovo_eeprom_noop, + .enable_resources = lenovo_eeprom_noop, + .init = lenovo_eeprom_init, +}; + +static void enable_dev(device_t dev) +{ + dev->ops = &lenovo_eeprom_operations; +} + +struct chip_operations drivers_i2c_lenovo_ops = { + CHIP_NAME("Lenovo EEPROM") + .enable_dev = enable_dev, +}; diff --git a/src/mainboard/lenovo/t60/devicetree.cb b/src/mainboard/lenovo/t60/devicetree.cb index 899ea5a..3e8d531 100644 --- a/src/mainboard/lenovo/t60/devicetree.cb +++ b/src/mainboard/lenovo/t60/devicetree.cb @@ -213,6 +213,31 @@ chip northbridge/intel/i945 register "reg11" = "0x07" device i2c 69 on end end + # eeprom, 8 virtual devices, same chip + chip drivers/i2c/lenovo + device i2c 54 on end + end + chip drivers/i2c/lenovo + device i2c 55 on end + end + chip drivers/i2c/lenovo + device i2c 56 on end + end + chip drivers/i2c/lenovo + device i2c 57 on end + end + chip drivers/i2c/lenovo + device i2c 5c on end + end + chip drivers/i2c/lenovo + device i2c 5d on end + end + chip drivers/i2c/lenovo + device i2c 5e on end + end + chip drivers/i2c/lenovo + device i2c 5f on end + end end end end diff --git a/src/mainboard/lenovo/x201/devicetree.cb b/src/mainboard/lenovo/x201/devicetree.cb index a8831c2..0502ad7 100644 --- a/src/mainboard/lenovo/x201/devicetree.cb +++ b/src/mainboard/lenovo/x201/devicetree.cb @@ -145,6 +145,31 @@ chip northbridge/intel/nehalem end device pci 1f.3 on # SMBUS subsystemid 0x17aa 0x2167 + # eeprom, 8 virtual devices, same chip + chip drivers/i2c/lenovo + device i2c 54 on end + end + chip drivers/i2c/lenovo + device i2c 55 on end + end + chip drivers/i2c/lenovo + device i2c 56 on end + end + chip drivers/i2c/lenovo + device i2c 57 on end + end + chip drivers/i2c/lenovo + device i2c 5c on end + end + chip drivers/i2c/lenovo + device i2c 5d on end + end + chip drivers/i2c/lenovo + device i2c 5e on end + end + chip drivers/i2c/lenovo + device i2c 5f on end + end end end end diff --git a/src/mainboard/lenovo/x230/devicetree.cb b/src/mainboard/lenovo/x230/devicetree.cb index 3650b38..5e12151 100644 --- a/src/mainboard/lenovo/x230/devicetree.cb +++ b/src/mainboard/lenovo/x230/devicetree.cb @@ -127,7 +127,33 @@ chip northbridge/intel/sandybridge end end # LPC bridge device pci 1f.2 on end # SATA Controller 1 - device pci 1f.3 on end # SMBus + device pci 1f.3 on + # eeprom, 8 virtual devices, same chip + chip drivers/i2c/lenovo + device i2c 54 on end + end + chip drivers/i2c/lenovo + device i2c 55 on end + end + chip drivers/i2c/lenovo + device i2c 56 on end + end + chip drivers/i2c/lenovo + device i2c 57 on end + end + chip drivers/i2c/lenovo + device i2c 5c on end + end + chip drivers/i2c/lenovo + device i2c 5d on end + end + chip drivers/i2c/lenovo + device i2c 5e on end + end + chip drivers/i2c/lenovo + device i2c 5f on end + end + end # SMBus device pci 1f.5 off end # SATA Controller 2 device pci 1f.6 on end # Thermal end diff --git a/src/mainboard/lenovo/x60/devicetree.cb b/src/mainboard/lenovo/x60/devicetree.cb index cfbb9db..5057f2e 100644 --- a/src/mainboard/lenovo/x60/devicetree.cb +++ b/src/mainboard/lenovo/x60/devicetree.cb @@ -190,7 +190,31 @@ chip northbridge/intel/i945 register "reg11" = "0x07" device i2c 69 on end end - + # eeprom, 8 virtual devices, same chip + chip drivers/i2c/lenovo + device i2c 54 on end + end + chip drivers/i2c/lenovo + device i2c 55 on end + end + chip drivers/i2c/lenovo + device i2c 56 on end + end + chip drivers/i2c/lenovo + device i2c 57 on end + end + chip drivers/i2c/lenovo + device i2c 5c on end + end + chip drivers/i2c/lenovo + device i2c 5d on end + end + chip drivers/i2c/lenovo + device i2c 5e on end + end + chip drivers/i2c/lenovo + device i2c 5f on end + end end end chip southbridge/ricoh/rl5c476