// SPDX-License-Identifier: GPL-2.0+ /* * Copyright (C) 2021 Sean Anderson */ #include #include #include #include #include #include /** * struct sysinfo_gpio_priv - GPIO sysinfo private data * @gpios: List of GPIOs used to detect the revision * @gpio_num: The number of GPIOs in @gpios * @revision: The revision as detected from the GPIOs. */ struct sysinfo_gpio_priv { struct gpio_desc *gpios; int gpio_num, revision; }; static int sysinfo_gpio_detect(struct udevice *dev) { int ret; struct sysinfo_gpio_priv *priv = dev_get_priv(dev); ret = dm_gpio_get_values_as_int_base3(priv->gpios, priv->gpio_num); if (ret < 0) return ret; priv->revision = ret; return 0; } static int sysinfo_gpio_get_int(struct udevice *dev, int id, int *val) { struct sysinfo_gpio_priv *priv = dev_get_priv(dev); switch (id) { case SYSINFO_ID_BOARD_MODEL: *val = priv->revision; return 0; default: return -EINVAL; }; } static int sysinfo_gpio_get_str(struct udevice *dev, int id, size_t size, char *val) { struct sysinfo_gpio_priv *priv = dev_get_priv(dev); switch (id) { case SYSINFO_ID_BOARD_MODEL: { const char *name = NULL; int i, ret; u32 revision; for (i = 0; i < priv->gpio_num; i++) { ret = dev_read_u32_index(dev, "revisions", i, &revision); if (ret) { if (ret != -EOVERFLOW) return ret; break; } if (revision == priv->revision) { ret = dev_read_string_index(dev, "names", i, &name); if (ret < 0) return ret; break; } } if (!name) name = "unknown"; strncpy(val, name, size); val[size - 1] = '\0'; return 0; } default: return -EINVAL; }; } static const struct sysinfo_ops sysinfo_gpio_ops = { .detect = sysinfo_gpio_detect, .get_int = sysinfo_gpio_get_int, .get_str = sysinfo_gpio_get_str, }; static int sysinfo_gpio_probe(struct udevice *dev) { int ret; struct sysinfo_gpio_priv *priv = dev_get_priv(dev); priv->gpio_num = gpio_get_list_count(dev, "gpios"); if (priv->gpio_num < 0) { dev_err(dev, "could not get gpios length (err = %d)\n", priv->gpio_num); return priv->gpio_num; } priv->gpios = calloc(priv->gpio_num, sizeof(*priv->gpios)); if (!priv->gpios) { dev_err(dev, "could not allocate memory for %d gpios\n", priv->gpio_num); return -ENOMEM; } ret = gpio_request_list_by_name(dev, "gpios", priv->gpios, priv->gpio_num, GPIOD_IS_IN); if (ret != priv->gpio_num) { dev_err(dev, "could not get gpios (err = %d)\n", priv->gpio_num); return ret; } if (!dev_read_bool(dev, "revisions") || !dev_read_bool(dev, "names")) { dev_err(dev, "revisions or names properties missing\n"); return -ENOENT; } return 0; } static const struct udevice_id sysinfo_gpio_ids[] = { { .compatible = "gpio-sysinfo" }, { /* sentinel */ } }; U_BOOT_DRIVER(sysinfo_gpio) = { .name = "sysinfo_gpio", .id = UCLASS_SYSINFO, .of_match = sysinfo_gpio_ids, .ops = &sysinfo_gpio_ops, .priv_auto = sizeof(struct sysinfo_gpio_priv), .probe = sysinfo_gpio_probe, };