treewide: remove redundant IS_ERR() before error code check
[linux/fpc-iii.git] / tools / iio / lsiio.c
blob2cf56fb2449ba0c3f67493561b25e9fc6d24e66c
1 // SPDX-License-Identifier: GPL-2.0-only
2 /*
3 * Industrial I/O utilities - lsiio.c
5 * Copyright (c) 2010 Manuel Stahl <manuel.stahl@iis.fraunhofer.de>
6 */
8 #include <string.h>
9 #include <dirent.h>
10 #include <stdio.h>
11 #include <errno.h>
12 #include <stdint.h>
13 #include <stdlib.h>
14 #include <unistd.h>
15 #include <sys/types.h>
16 #include <sys/stat.h>
17 #include <sys/dir.h>
18 #include "iio_utils.h"
20 static enum verbosity {
21 VERBLEVEL_DEFAULT, /* 0 gives lspci behaviour */
22 VERBLEVEL_SENSORS, /* 1 lists sensors */
23 } verblevel = VERBLEVEL_DEFAULT;
25 const char *type_device = "iio:device";
26 const char *type_trigger = "trigger";
28 static inline int check_prefix(const char *str, const char *prefix)
30 return strlen(str) > strlen(prefix) &&
31 strncmp(str, prefix, strlen(prefix)) == 0;
34 static inline int check_postfix(const char *str, const char *postfix)
36 return strlen(str) > strlen(postfix) &&
37 strcmp(str + strlen(str) - strlen(postfix), postfix) == 0;
40 static int dump_channels(const char *dev_dir_name)
42 DIR *dp;
43 const struct dirent *ent;
45 dp = opendir(dev_dir_name);
46 if (!dp)
47 return -errno;
49 while (ent = readdir(dp), ent)
50 if (check_prefix(ent->d_name, "in_") &&
51 (check_postfix(ent->d_name, "_raw") ||
52 check_postfix(ent->d_name, "_input")))
53 printf(" %-10s\n", ent->d_name);
55 return (closedir(dp) == -1) ? -errno : 0;
58 static int dump_one_device(const char *dev_dir_name)
60 char name[IIO_MAX_NAME_LENGTH];
61 int dev_idx;
62 int ret;
64 ret = sscanf(dev_dir_name + strlen(iio_dir) + strlen(type_device), "%i",
65 &dev_idx);
66 if (ret != 1)
67 return -EINVAL;
69 ret = read_sysfs_string("name", dev_dir_name, name);
70 if (ret < 0)
71 return ret;
73 printf("Device %03d: %s\n", dev_idx, name);
75 if (verblevel >= VERBLEVEL_SENSORS)
76 return dump_channels(dev_dir_name);
78 return 0;
81 static int dump_one_trigger(const char *dev_dir_name)
83 char name[IIO_MAX_NAME_LENGTH];
84 int dev_idx;
85 int ret;
87 ret = sscanf(dev_dir_name + strlen(iio_dir) + strlen(type_trigger),
88 "%i", &dev_idx);
89 if (ret != 1)
90 return -EINVAL;
92 ret = read_sysfs_string("name", dev_dir_name, name);
93 if (ret < 0)
94 return ret;
96 printf("Trigger %03d: %s\n", dev_idx, name);
98 return 0;
101 static int dump_devices(void)
103 const struct dirent *ent;
104 int ret;
105 DIR *dp;
107 dp = opendir(iio_dir);
108 if (!dp) {
109 fprintf(stderr, "No industrial I/O devices available\n");
110 return -ENODEV;
113 while (ent = readdir(dp), ent) {
114 if (check_prefix(ent->d_name, type_device)) {
115 char *dev_dir_name;
117 if (asprintf(&dev_dir_name, "%s%s", iio_dir,
118 ent->d_name) < 0) {
119 ret = -ENOMEM;
120 goto error_close_dir;
123 ret = dump_one_device(dev_dir_name);
124 if (ret) {
125 free(dev_dir_name);
126 goto error_close_dir;
129 free(dev_dir_name);
130 if (verblevel >= VERBLEVEL_SENSORS)
131 printf("\n");
134 rewinddir(dp);
135 while (ent = readdir(dp), ent) {
136 if (check_prefix(ent->d_name, type_trigger)) {
137 char *dev_dir_name;
139 if (asprintf(&dev_dir_name, "%s%s", iio_dir,
140 ent->d_name) < 0) {
141 ret = -ENOMEM;
142 goto error_close_dir;
145 ret = dump_one_trigger(dev_dir_name);
146 if (ret) {
147 free(dev_dir_name);
148 goto error_close_dir;
151 free(dev_dir_name);
155 return (closedir(dp) == -1) ? -errno : 0;
157 error_close_dir:
158 if (closedir(dp) == -1)
159 perror("dump_devices(): Failed to close directory");
161 return ret;
164 int main(int argc, char **argv)
166 int c, err = 0;
168 while ((c = getopt(argc, argv, "v")) != EOF) {
169 switch (c) {
170 case 'v':
171 verblevel++;
172 break;
174 case '?':
175 default:
176 err++;
177 break;
180 if (err || argc > optind) {
181 fprintf(stderr, "Usage: lsiio [options]...\n"
182 "List industrial I/O devices\n"
183 " -v Increase verbosity (may be given multiple times)\n");
184 exit(1);
187 return dump_devices();