]> git.ipfire.org Git - thirdparty/u-boot.git/blobdiff - cmd/gpio.c
cmd: sbi: formatting PolarFire Hart Software Services version
[thirdparty/u-boot.git] / cmd / gpio.c
index c60946bc06934cdf71d03029dd5a4121993a8b3c..dab6f7097aec1a0c9ea40a64d128dfa665010377 100644 (file)
 #include <command.h>
 #include <errno.h>
 #include <dm.h>
+#include <log.h>
+#include <malloc.h>
+#ifdef CONFIG_CMD_GPIO_READ
+#include <env.h>
+#endif
 #include <asm/gpio.h>
+#include <linux/err.h>
+#include <dm/device_compat.h>
 
 __weak int name_to_gpio(const char *name)
 {
-       return simple_strtoul(name, NULL, 10);
+       return dectoul(name, NULL);
 }
 
 enum gpio_cmd {
-       GPIO_INPUT,
-       GPIO_SET,
-       GPIO_CLEAR,
-       GPIO_TOGGLE,
+       GPIOC_INPUT,
+       GPIOC_SET,
+       GPIOC_CLEAR,
+       GPIOC_TOGGLE,
+#ifdef CONFIG_CMD_GPIO_READ
+       GPIOC_READ,
+#endif
 };
 
 #if defined(CONFIG_DM_GPIO) && !defined(gpio_status)
@@ -34,7 +44,7 @@ enum {
 };
 
 static void gpio_get_description(struct udevice *dev, const char *bank_name,
-                                int offset, int *flagsp)
+                                int offset, int *flagsp, bool show_all)
 {
        char buf[80];
        int ret;
@@ -42,7 +52,7 @@ static void gpio_get_description(struct udevice *dev, const char *bank_name,
        ret = gpio_get_function(dev, offset, NULL);
        if (ret < 0)
                goto err;
-       if (!(*flagsp & FLAG_SHOW_ALL) && ret == GPIOF_UNUSED)
+       if (!show_all && !(*flagsp & FLAG_SHOW_ALL) && ret == GPIOF_UNUSED)
                return;
        if ((*flagsp & FLAG_SHOW_BANK) && bank_name) {
                if (*flagsp & FLAG_SHOW_NEWLINE) {
@@ -60,7 +70,8 @@ static void gpio_get_description(struct udevice *dev, const char *bank_name,
        printf("%s\n", buf);
        return;
 err:
-       printf("Error %d\n", ret);
+       if (ret != -ENOENT)
+               printf("Error %d\n", ret);
 }
 
 static int do_gpio_status(bool all, const char *gpio_name)
@@ -68,17 +79,24 @@ static int do_gpio_status(bool all, const char *gpio_name)
        struct udevice *dev;
        int banklen;
        int flags;
-       int ret;
+       int ret, err = 0;
 
        flags = 0;
        if (gpio_name && !*gpio_name)
                gpio_name = NULL;
-       for (ret = uclass_first_device(UCLASS_GPIO, &dev);
+       for (ret = uclass_first_device_check(UCLASS_GPIO, &dev);
             dev;
-            ret = uclass_next_device(&dev)) {
+            ret = uclass_next_device_check(&dev)) {
                const char *bank_name;
                int num_bits;
 
+               if (ret) {
+                       printf("GPIO device %s probe error %i\n",
+                              dev->name, ret);
+                       err = ret;
+                       continue;
+               }
+
                flags |= FLAG_SHOW_BANK;
                if (all)
                        flags |= FLAG_SHOW_ALL;
@@ -90,19 +108,19 @@ static int do_gpio_status(bool all, const char *gpio_name)
                banklen = bank_name ? strlen(bank_name) : 0;
 
                if (!gpio_name || !bank_name ||
-                   !strncmp(gpio_name, bank_name, banklen)) {
-                       const char *p = NULL;
+                   !strncasecmp(gpio_name, bank_name, banklen)) {
+                       const char *p;
                        int offset;
 
                        p = gpio_name + banklen;
                        if (gpio_name && *p) {
-                               offset = simple_strtoul(p, NULL, 10);
+                               offset = dectoul(p, NULL);
                                gpio_get_description(dev, bank_name, offset,
-                                                    &flags);
+                                                    &flags, true);
                        } else {
                                for (offset = 0; offset < num_bits; offset++) {
                                        gpio_get_description(dev, bank_name,
-                                                            offset, &flags);
+                                                    offset, &flags, false);
                                }
                        }
                }
@@ -111,16 +129,20 @@ static int do_gpio_status(bool all, const char *gpio_name)
                        flags |= FLAG_SHOW_NEWLINE;
        }
 
-       return ret;
+       return err;
 }
 #endif
 
-static int do_gpio(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
+static int do_gpio(struct cmd_tbl *cmdtp, int flag, int argc,
+                  char *const argv[])
 {
        unsigned int gpio;
        enum gpio_cmd sub_cmd;
        int value;
        const char *str_cmd, *str_gpio = NULL;
+#ifdef CONFIG_CMD_GPIO_READ
+       const char *str_var = NULL;
+#endif
        int ret;
 #ifdef CONFIG_DM_GPIO
        bool all = false;
@@ -133,11 +155,20 @@ static int do_gpio(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
        argc -= 2;
        argv += 2;
 #ifdef CONFIG_DM_GPIO
-       if (argc > 0 && !strcmp(*argv, "-a")) {
+       if (argc > 0 && !strncmp(str_cmd, "status", 2) && !strcmp(*argv, "-a")) {
                all = true;
                argc--;
                argv++;
        }
+#endif
+#ifdef CONFIG_CMD_GPIO_READ
+       if (argc > 0 && !strncmp(str_cmd, "read", 2)) {
+               if (argc < 2)
+                       goto show_usage;
+               str_var = *argv;
+               argc--;
+               argv++;
+       }
 #endif
        if (argc > 0)
                str_gpio = *argv;
@@ -158,11 +189,25 @@ static int do_gpio(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
 
        /* parse the behavior */
        switch (*str_cmd) {
-               case 'i': sub_cmd = GPIO_INPUT;  break;
-               case 's': sub_cmd = GPIO_SET;    break;
-               case 'c': sub_cmd = GPIO_CLEAR;  break;
-               case 't': sub_cmd = GPIO_TOGGLE; break;
-               default:  goto show_usage;
+       case 'i':
+               sub_cmd = GPIOC_INPUT;
+               break;
+       case 's':
+               sub_cmd = GPIOC_SET;
+               break;
+       case 'c':
+               sub_cmd = GPIOC_CLEAR;
+               break;
+       case 't':
+               sub_cmd = GPIOC_TOGGLE;
+               break;
+#ifdef CONFIG_CMD_GPIO_READ
+       case 'r':
+               sub_cmd = GPIOC_READ;
+               break;
+#endif
+       default:
+               goto show_usage;
        }
 
 #if defined(CONFIG_DM_GPIO)
@@ -192,18 +237,22 @@ static int do_gpio(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
        }
 
        /* finally, let's do it: set direction and exec command */
-       if (sub_cmd == GPIO_INPUT) {
+       if (sub_cmd == GPIOC_INPUT
+#ifdef CONFIG_CMD_GPIO_READ
+                       || sub_cmd == GPIOC_READ
+#endif
+                       ) {
                gpio_direction_input(gpio);
                value = gpio_get_value(gpio);
        } else {
                switch (sub_cmd) {
-               case GPIO_SET:
+               case GPIOC_SET:
                        value = 1;
                        break;
-               case GPIO_CLEAR:
+               case GPIOC_CLEAR:
                        value = 0;
                        break;
-               case GPIO_TOGGLE:
+               case GPIOC_TOGGLE:
                        value = gpio_get_value(gpio);
                        if (!IS_ERR_VALUE(value))
                                value = !value;
@@ -214,27 +263,56 @@ static int do_gpio(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
                gpio_direction_output(gpio, value);
        }
        printf("gpio: pin %s (gpio %u) value is ", str_gpio, gpio);
-       if (IS_ERR_VALUE(value))
+
+       if (IS_ERR_VALUE(value)) {
                printf("unknown (ret=%d)\n", value);
-       else
+               goto err;
+       } else {
                printf("%d\n", value);
-       if (sub_cmd != GPIO_INPUT && !IS_ERR_VALUE(value)) {
+#ifdef CONFIG_CMD_GPIO_READ
+               if (sub_cmd == GPIOC_READ)
+                       env_set_ulong(str_var, (ulong)value);
+#endif
+       }
+
+       if (sub_cmd != GPIOC_INPUT && !IS_ERR_VALUE(value)
+#ifdef CONFIG_CMD_GPIO_READ
+                       && sub_cmd != GPIOC_READ
+#endif
+                       ) {
                int nval = gpio_get_value(gpio);
 
-               if (IS_ERR_VALUE(nval))
+               if (IS_ERR_VALUE(nval)) {
                        printf("   Warning: no access to GPIO output value\n");
-               else if (nval != value)
+                       goto err;
+               } else if (nval != value) {
                        printf("   Warning: value of pin is still %d\n", nval);
+                       goto err;
+               }
        }
 
        if (ret != -EBUSY)
                gpio_free(gpio);
 
-       return value;
+       /*
+        * Whilst wrong, the legacy gpio input command returns the pin
+        * value, or CMD_RET_FAILURE (which is indistinguishable from a
+        * valid pin value).
+        */
+       return (sub_cmd == GPIOC_INPUT) ? value : CMD_RET_SUCCESS;
+
+err:
+       if (ret != -EBUSY)
+               gpio_free(gpio);
+       return CMD_RET_FAILURE;
 }
 
 U_BOOT_CMD(gpio, 4, 0, do_gpio,
           "query and control gpio pins",
           "<input|set|clear|toggle> <pin>\n"
           "    - input/set/clear/toggle the specified pin\n"
+#ifdef CONFIG_CMD_GPIO_READ
+          "gpio read <name> <pin>\n"
+          "    - set environment variable 'name' to the specified pin\n"
+#endif
           "gpio status [-a] [<bank> | <pin>]  - show [all/claimed] GPIOs");