xref: /kernel/linux/linux-6.6/tools/gpio/lsgpio.c (revision 62306a36)
162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-only
262306a36Sopenharmony_ci/*
362306a36Sopenharmony_ci * lsgpio - example on how to list the GPIO lines on a system
462306a36Sopenharmony_ci *
562306a36Sopenharmony_ci * Copyright (C) 2015 Linus Walleij
662306a36Sopenharmony_ci *
762306a36Sopenharmony_ci * Usage:
862306a36Sopenharmony_ci *	lsgpio <-n device-name>
962306a36Sopenharmony_ci */
1062306a36Sopenharmony_ci
1162306a36Sopenharmony_ci#include <unistd.h>
1262306a36Sopenharmony_ci#include <stdlib.h>
1362306a36Sopenharmony_ci#include <stdbool.h>
1462306a36Sopenharmony_ci#include <stdio.h>
1562306a36Sopenharmony_ci#include <dirent.h>
1662306a36Sopenharmony_ci#include <errno.h>
1762306a36Sopenharmony_ci#include <string.h>
1862306a36Sopenharmony_ci#include <poll.h>
1962306a36Sopenharmony_ci#include <fcntl.h>
2062306a36Sopenharmony_ci#include <getopt.h>
2162306a36Sopenharmony_ci#include <sys/ioctl.h>
2262306a36Sopenharmony_ci#include <linux/gpio.h>
2362306a36Sopenharmony_ci
2462306a36Sopenharmony_ci#include "gpio-utils.h"
2562306a36Sopenharmony_ci
2662306a36Sopenharmony_cistruct gpio_flag {
2762306a36Sopenharmony_ci	char *name;
2862306a36Sopenharmony_ci	unsigned long long mask;
2962306a36Sopenharmony_ci};
3062306a36Sopenharmony_ci
3162306a36Sopenharmony_cistruct gpio_flag flagnames[] = {
3262306a36Sopenharmony_ci	{
3362306a36Sopenharmony_ci		.name = "used",
3462306a36Sopenharmony_ci		.mask = GPIO_V2_LINE_FLAG_USED,
3562306a36Sopenharmony_ci	},
3662306a36Sopenharmony_ci	{
3762306a36Sopenharmony_ci		.name = "input",
3862306a36Sopenharmony_ci		.mask = GPIO_V2_LINE_FLAG_INPUT,
3962306a36Sopenharmony_ci	},
4062306a36Sopenharmony_ci	{
4162306a36Sopenharmony_ci		.name = "output",
4262306a36Sopenharmony_ci		.mask = GPIO_V2_LINE_FLAG_OUTPUT,
4362306a36Sopenharmony_ci	},
4462306a36Sopenharmony_ci	{
4562306a36Sopenharmony_ci		.name = "active-low",
4662306a36Sopenharmony_ci		.mask = GPIO_V2_LINE_FLAG_ACTIVE_LOW,
4762306a36Sopenharmony_ci	},
4862306a36Sopenharmony_ci	{
4962306a36Sopenharmony_ci		.name = "open-drain",
5062306a36Sopenharmony_ci		.mask = GPIO_V2_LINE_FLAG_OPEN_DRAIN,
5162306a36Sopenharmony_ci	},
5262306a36Sopenharmony_ci	{
5362306a36Sopenharmony_ci		.name = "open-source",
5462306a36Sopenharmony_ci		.mask = GPIO_V2_LINE_FLAG_OPEN_SOURCE,
5562306a36Sopenharmony_ci	},
5662306a36Sopenharmony_ci	{
5762306a36Sopenharmony_ci		.name = "pull-up",
5862306a36Sopenharmony_ci		.mask = GPIO_V2_LINE_FLAG_BIAS_PULL_UP,
5962306a36Sopenharmony_ci	},
6062306a36Sopenharmony_ci	{
6162306a36Sopenharmony_ci		.name = "pull-down",
6262306a36Sopenharmony_ci		.mask = GPIO_V2_LINE_FLAG_BIAS_PULL_DOWN,
6362306a36Sopenharmony_ci	},
6462306a36Sopenharmony_ci	{
6562306a36Sopenharmony_ci		.name = "bias-disabled",
6662306a36Sopenharmony_ci		.mask = GPIO_V2_LINE_FLAG_BIAS_DISABLED,
6762306a36Sopenharmony_ci	},
6862306a36Sopenharmony_ci	{
6962306a36Sopenharmony_ci		.name = "clock-realtime",
7062306a36Sopenharmony_ci		.mask = GPIO_V2_LINE_FLAG_EVENT_CLOCK_REALTIME,
7162306a36Sopenharmony_ci	},
7262306a36Sopenharmony_ci};
7362306a36Sopenharmony_ci
7462306a36Sopenharmony_cistatic void print_attributes(struct gpio_v2_line_info *info)
7562306a36Sopenharmony_ci{
7662306a36Sopenharmony_ci	int i;
7762306a36Sopenharmony_ci	const char *field_format = "%s";
7862306a36Sopenharmony_ci
7962306a36Sopenharmony_ci	for (i = 0; i < ARRAY_SIZE(flagnames); i++) {
8062306a36Sopenharmony_ci		if (info->flags & flagnames[i].mask) {
8162306a36Sopenharmony_ci			fprintf(stdout, field_format, flagnames[i].name);
8262306a36Sopenharmony_ci			field_format = ", %s";
8362306a36Sopenharmony_ci		}
8462306a36Sopenharmony_ci	}
8562306a36Sopenharmony_ci
8662306a36Sopenharmony_ci	if ((info->flags & GPIO_V2_LINE_FLAG_EDGE_RISING) &&
8762306a36Sopenharmony_ci	    (info->flags & GPIO_V2_LINE_FLAG_EDGE_FALLING))
8862306a36Sopenharmony_ci		fprintf(stdout, field_format, "both-edges");
8962306a36Sopenharmony_ci	else if (info->flags & GPIO_V2_LINE_FLAG_EDGE_RISING)
9062306a36Sopenharmony_ci		fprintf(stdout, field_format, "rising-edge");
9162306a36Sopenharmony_ci	else if (info->flags & GPIO_V2_LINE_FLAG_EDGE_FALLING)
9262306a36Sopenharmony_ci		fprintf(stdout, field_format, "falling-edge");
9362306a36Sopenharmony_ci
9462306a36Sopenharmony_ci	for (i = 0; i < info->num_attrs; i++) {
9562306a36Sopenharmony_ci		if (info->attrs[i].id == GPIO_V2_LINE_ATTR_ID_DEBOUNCE)
9662306a36Sopenharmony_ci			fprintf(stdout, ", debounce_period=%dusec",
9762306a36Sopenharmony_ci				info->attrs[i].debounce_period_us);
9862306a36Sopenharmony_ci	}
9962306a36Sopenharmony_ci}
10062306a36Sopenharmony_ci
10162306a36Sopenharmony_ciint list_device(const char *device_name)
10262306a36Sopenharmony_ci{
10362306a36Sopenharmony_ci	struct gpiochip_info cinfo;
10462306a36Sopenharmony_ci	char *chrdev_name;
10562306a36Sopenharmony_ci	int fd;
10662306a36Sopenharmony_ci	int ret;
10762306a36Sopenharmony_ci	int i;
10862306a36Sopenharmony_ci
10962306a36Sopenharmony_ci	ret = asprintf(&chrdev_name, "/dev/%s", device_name);
11062306a36Sopenharmony_ci	if (ret < 0)
11162306a36Sopenharmony_ci		return -ENOMEM;
11262306a36Sopenharmony_ci
11362306a36Sopenharmony_ci	fd = open(chrdev_name, 0);
11462306a36Sopenharmony_ci	if (fd == -1) {
11562306a36Sopenharmony_ci		ret = -errno;
11662306a36Sopenharmony_ci		fprintf(stderr, "Failed to open %s\n", chrdev_name);
11762306a36Sopenharmony_ci		goto exit_free_name;
11862306a36Sopenharmony_ci	}
11962306a36Sopenharmony_ci
12062306a36Sopenharmony_ci	/* Inspect this GPIO chip */
12162306a36Sopenharmony_ci	ret = ioctl(fd, GPIO_GET_CHIPINFO_IOCTL, &cinfo);
12262306a36Sopenharmony_ci	if (ret == -1) {
12362306a36Sopenharmony_ci		ret = -errno;
12462306a36Sopenharmony_ci		perror("Failed to issue CHIPINFO IOCTL\n");
12562306a36Sopenharmony_ci		goto exit_close_error;
12662306a36Sopenharmony_ci	}
12762306a36Sopenharmony_ci	fprintf(stdout, "GPIO chip: %s, \"%s\", %u GPIO lines\n",
12862306a36Sopenharmony_ci		cinfo.name, cinfo.label, cinfo.lines);
12962306a36Sopenharmony_ci
13062306a36Sopenharmony_ci	/* Loop over the lines and print info */
13162306a36Sopenharmony_ci	for (i = 0; i < cinfo.lines; i++) {
13262306a36Sopenharmony_ci		struct gpio_v2_line_info linfo;
13362306a36Sopenharmony_ci
13462306a36Sopenharmony_ci		memset(&linfo, 0, sizeof(linfo));
13562306a36Sopenharmony_ci		linfo.offset = i;
13662306a36Sopenharmony_ci
13762306a36Sopenharmony_ci		ret = ioctl(fd, GPIO_V2_GET_LINEINFO_IOCTL, &linfo);
13862306a36Sopenharmony_ci		if (ret == -1) {
13962306a36Sopenharmony_ci			ret = -errno;
14062306a36Sopenharmony_ci			perror("Failed to issue LINEINFO IOCTL\n");
14162306a36Sopenharmony_ci			goto exit_close_error;
14262306a36Sopenharmony_ci		}
14362306a36Sopenharmony_ci		fprintf(stdout, "\tline %2d:", linfo.offset);
14462306a36Sopenharmony_ci		if (linfo.name[0])
14562306a36Sopenharmony_ci			fprintf(stdout, " \"%s\"", linfo.name);
14662306a36Sopenharmony_ci		else
14762306a36Sopenharmony_ci			fprintf(stdout, " unnamed");
14862306a36Sopenharmony_ci		if (linfo.consumer[0])
14962306a36Sopenharmony_ci			fprintf(stdout, " \"%s\"", linfo.consumer);
15062306a36Sopenharmony_ci		else
15162306a36Sopenharmony_ci			fprintf(stdout, " unused");
15262306a36Sopenharmony_ci		if (linfo.flags) {
15362306a36Sopenharmony_ci			fprintf(stdout, " [");
15462306a36Sopenharmony_ci			print_attributes(&linfo);
15562306a36Sopenharmony_ci			fprintf(stdout, "]");
15662306a36Sopenharmony_ci		}
15762306a36Sopenharmony_ci		fprintf(stdout, "\n");
15862306a36Sopenharmony_ci
15962306a36Sopenharmony_ci	}
16062306a36Sopenharmony_ci
16162306a36Sopenharmony_ciexit_close_error:
16262306a36Sopenharmony_ci	if (close(fd) == -1)
16362306a36Sopenharmony_ci		perror("Failed to close GPIO character device file");
16462306a36Sopenharmony_ciexit_free_name:
16562306a36Sopenharmony_ci	free(chrdev_name);
16662306a36Sopenharmony_ci	return ret;
16762306a36Sopenharmony_ci}
16862306a36Sopenharmony_ci
16962306a36Sopenharmony_civoid print_usage(void)
17062306a36Sopenharmony_ci{
17162306a36Sopenharmony_ci	fprintf(stderr, "Usage: lsgpio [options]...\n"
17262306a36Sopenharmony_ci		"List GPIO chips, lines and states\n"
17362306a36Sopenharmony_ci		"  -n <name>  List GPIOs on a named device\n"
17462306a36Sopenharmony_ci		"  -?         This helptext\n"
17562306a36Sopenharmony_ci	);
17662306a36Sopenharmony_ci}
17762306a36Sopenharmony_ci
17862306a36Sopenharmony_ciint main(int argc, char **argv)
17962306a36Sopenharmony_ci{
18062306a36Sopenharmony_ci	const char *device_name = NULL;
18162306a36Sopenharmony_ci	int ret;
18262306a36Sopenharmony_ci	int c;
18362306a36Sopenharmony_ci
18462306a36Sopenharmony_ci	while ((c = getopt(argc, argv, "n:")) != -1) {
18562306a36Sopenharmony_ci		switch (c) {
18662306a36Sopenharmony_ci		case 'n':
18762306a36Sopenharmony_ci			device_name = optarg;
18862306a36Sopenharmony_ci			break;
18962306a36Sopenharmony_ci		case '?':
19062306a36Sopenharmony_ci			print_usage();
19162306a36Sopenharmony_ci			return -1;
19262306a36Sopenharmony_ci		}
19362306a36Sopenharmony_ci	}
19462306a36Sopenharmony_ci
19562306a36Sopenharmony_ci	if (device_name)
19662306a36Sopenharmony_ci		ret = list_device(device_name);
19762306a36Sopenharmony_ci	else {
19862306a36Sopenharmony_ci		const struct dirent *ent;
19962306a36Sopenharmony_ci		DIR *dp;
20062306a36Sopenharmony_ci
20162306a36Sopenharmony_ci		/* List all GPIO devices one at a time */
20262306a36Sopenharmony_ci		dp = opendir("/dev");
20362306a36Sopenharmony_ci		if (!dp) {
20462306a36Sopenharmony_ci			ret = -errno;
20562306a36Sopenharmony_ci			goto error_out;
20662306a36Sopenharmony_ci		}
20762306a36Sopenharmony_ci
20862306a36Sopenharmony_ci		ret = -ENOENT;
20962306a36Sopenharmony_ci		while (ent = readdir(dp), ent) {
21062306a36Sopenharmony_ci			if (check_prefix(ent->d_name, "gpiochip")) {
21162306a36Sopenharmony_ci				ret = list_device(ent->d_name);
21262306a36Sopenharmony_ci				if (ret)
21362306a36Sopenharmony_ci					break;
21462306a36Sopenharmony_ci			}
21562306a36Sopenharmony_ci		}
21662306a36Sopenharmony_ci
21762306a36Sopenharmony_ci		ret = 0;
21862306a36Sopenharmony_ci		if (closedir(dp) == -1) {
21962306a36Sopenharmony_ci			perror("scanning devices: Failed to close directory");
22062306a36Sopenharmony_ci			ret = -errno;
22162306a36Sopenharmony_ci		}
22262306a36Sopenharmony_ci	}
22362306a36Sopenharmony_cierror_out:
22462306a36Sopenharmony_ci	return ret;
22562306a36Sopenharmony_ci}
226