// SPDX-License-Identifier: GPL-2.0+ /* * Copyright (C) 2021 Marek Vasut */ #include #include #include #include #include #define BOARD_CODE_MASK 0xF8 #define BOARD_REV_MASK 0x07 #define BOARD_CODE_SHIFT 0x03 #define BOARD_SALVATOR_X 0x0 #define BOARD_KRIEK 0x1 #define BOARD_STARTER_KIT 0x2 #define BOARD_EAGLE 0x3 #define BOARD_SALVATOR_XS 0x4 #define BOARD_CONDOR 0x6 #define BOARD_DRAAK 0x7 #define BOARD_EBISU 0x8 #define BOARD_STARTER_KIT_PRE 0xB #define BOARD_EBISU_4D 0xD #define BOARD_CONDOR_I 0x10 /** * struct sysinfo_rcar_priv - sysinfo private data * @boardname: board model and revision * @val: board ID value from eeprom */ struct sysinfo_rcar_priv { char boardmodel[64]; u8 val; }; static int sysinfo_rcar_detect(struct udevice *dev) { struct sysinfo_rcar_priv *priv = dev_get_priv(dev); return priv->val == 0xff; } static int sysinfo_rcar_get_str(struct udevice *dev, int id, size_t size, char *val) { struct sysinfo_rcar_priv *priv = dev_get_priv(dev); switch (id) { case SYSINFO_ID_BOARD_MODEL: strncpy(val, priv->boardmodel, size); val[size - 1] = '\0'; return 0; default: return -EINVAL; }; } static const struct sysinfo_ops sysinfo_rcar_ops = { .detect = sysinfo_rcar_detect, .get_str = sysinfo_rcar_get_str, }; static void sysinfo_rcar_parse(struct sysinfo_rcar_priv *priv) { const u8 board_id = (priv->val & BOARD_CODE_MASK) >> BOARD_CODE_SHIFT; const u8 board_rev = priv->val & BOARD_REV_MASK; bool salvator_xs = false; bool ebisu_4d = false; bool condor_i = false; char rev_major = '?'; char rev_minor = '?'; switch (board_id) { case BOARD_SALVATOR_XS: salvator_xs = true; fallthrough; case BOARD_SALVATOR_X: if (!(board_rev & ~1)) { /* Only rev 0 and 1 is valid */ rev_major = '1'; rev_minor = '0' + (board_rev & BIT(0)); } snprintf(priv->boardmodel, sizeof(priv->boardmodel), "Renesas Salvator-X%s board rev %c.%c", salvator_xs ? "S" : "", rev_major, rev_minor); return; case BOARD_STARTER_KIT: if (!(board_rev & ~1)) { /* Only rev 0 and 1 is valid */ rev_major = (board_rev & BIT(0)) ? '3' : '1'; rev_minor = '0'; } snprintf(priv->boardmodel, sizeof(priv->boardmodel), "Renesas Starter Kit board rev %c.%c", rev_major, rev_minor); return; case BOARD_STARTER_KIT_PRE: if (!(board_rev & ~3)) { /* Only rev 0..3 is valid */ rev_major = (board_rev & BIT(1)) ? '2' : '1'; rev_minor = (board_rev == 3) ? '1' : '0'; } snprintf(priv->boardmodel, sizeof(priv->boardmodel), "Renesas Starter Kit Premier board rev %c.%c", rev_major, rev_minor); return; case BOARD_EAGLE: if (!board_rev) { /* Only rev 0 is valid */ rev_major = '1'; rev_minor = '0'; } snprintf(priv->boardmodel, sizeof(priv->boardmodel), "Renesas Eagle board rev %c.%c", rev_major, rev_minor); return; case BOARD_EBISU_4D: ebisu_4d = true; fallthrough; case BOARD_EBISU: if (!board_rev) { /* Only rev 0 is valid */ rev_major = '1'; rev_minor = '0'; } snprintf(priv->boardmodel, sizeof(priv->boardmodel), "Renesas Ebisu%s board rev %c.%c", ebisu_4d ? "-4D" : "", rev_major, rev_minor); return; case BOARD_DRAAK: if (!board_rev) { /* Only rev 0 is valid */ rev_major = '1'; rev_minor = '0'; } snprintf(priv->boardmodel, sizeof(priv->boardmodel), "Renesas Draak board rev %c.%c", rev_major, rev_minor); return; case BOARD_KRIEK: if (!board_rev) { /* Only rev 0 is valid */ rev_major = '1'; rev_minor = '0'; } snprintf(priv->boardmodel, sizeof(priv->boardmodel), "Renesas Kriek board rev %c.%c", rev_major, rev_minor); return; case BOARD_CONDOR_I: condor_i = true; fallthrough; case BOARD_CONDOR: if (!board_rev) { /* Only rev 0 is valid */ rev_major = '1'; rev_minor = '0'; } snprintf(priv->boardmodel, sizeof(priv->boardmodel), "Renesas Condor%s board rev %c.%c", condor_i ? "-I" : "", rev_major, rev_minor); return; default: snprintf(priv->boardmodel, sizeof(priv->boardmodel), "Renesas -Unknown- board rev ?.?"); priv->val = 0xff; return; } } static int sysinfo_rcar_probe(struct udevice *dev) { struct sysinfo_rcar_priv *priv = dev_get_priv(dev); struct ofnode_phandle_args phandle_args; struct udevice *i2c_eeprom; u32 offset; int ret; offset = dev_read_u32_default(dev, "offset", 0x70); ret = dev_read_phandle_with_args(dev, "i2c-eeprom", NULL, 0, 0, &phandle_args); if (ret) { debug("%s: i2c-eeprom backing device not specified\n", dev->name); return ret; } ret = uclass_get_device_by_ofnode(UCLASS_I2C_EEPROM, phandle_args.node, &i2c_eeprom); if (ret) { debug("%s: could not get backing device\n", dev->name); return ret; } ret = i2c_eeprom_read(i2c_eeprom, offset, &priv->val, 1); if (ret < 0) { debug("%s: read failed\n", __func__); return -EIO; } sysinfo_rcar_parse(priv); return 0; } static const struct udevice_id sysinfo_rcar_ids[] = { { .compatible = "renesas,rcar-sysinfo" }, { /* sentinel */ } }; U_BOOT_DRIVER(sysinfo_rcar) = { .name = "sysinfo_rcar", .id = UCLASS_SYSINFO, .of_match = sysinfo_rcar_ids, .ops = &sysinfo_rcar_ops, .priv_auto = sizeof(struct sysinfo_rcar_priv), .probe = sysinfo_rcar_probe, };