123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307 |
- /*
- * Copyright (c) 2006-2023, RT-Thread Development Team
- *
- * SPDX-License-Identifier: Apache-2.0
- *
- * Change Logs:
- * Date Author Notes
- */
- #include "proc.h"
- #include "procfs.h"
- #include <rthw.h>
- #include <rtdbg.h>
- #include <fcntl.h>
- #include <errno.h>
- #include <dfs_dentry.h>
- #include <rthw.h>
- #include <rtthread.h>
- #include <string.h>
- #define LIST_FIND_OBJ_NR 8
- struct device_show
- {
- char *buf;
- int size;
- int len;
- int index;
- };
- typedef struct
- {
- rt_list_t *list;
- rt_list_t **array;
- rt_uint8_t type;
- int nr; /* input: max nr, can't be 0 */
- int nr_out; /* out: got nr */
- } list_get_next_t;
- static void list_find_init(list_get_next_t *p, rt_uint8_t type, rt_list_t **array, int nr)
- {
- struct rt_object_information *info;
- rt_list_t *list;
- info = rt_object_get_information((enum rt_object_class_type)type);
- list = &info->object_list;
- p->list = list;
- p->type = type;
- p->array = array;
- p->nr = nr;
- p->nr_out = 0;
- }
- static rt_list_t *list_get_next(rt_list_t *current, list_get_next_t *arg)
- {
- int first_flag = 0;
- rt_base_t level;
- rt_list_t *node, *list;
- rt_list_t **array;
- struct rt_object_information *info;
- int nr;
- arg->nr_out = 0;
- if (!arg->nr || !arg->type)
- {
- return (rt_list_t *)RT_NULL;
- }
- list = arg->list;
- info = rt_list_entry(list, struct rt_object_information, object_list);
- if (!current) /* find first */
- {
- node = list;
- first_flag = 1;
- }
- else
- {
- node = current;
- }
- level = rt_spin_lock_irqsave(&info->spinlock);
- if (!first_flag)
- {
- struct rt_object *obj;
- /* The node in the list? */
- obj = rt_list_entry(node, struct rt_object, list);
- if ((obj->type & ~RT_Object_Class_Static) != arg->type)
- {
- rt_spin_unlock_irqrestore(&info->spinlock, level);
- return (rt_list_t *)RT_NULL;
- }
- }
- nr = 0;
- array = arg->array;
- while (1)
- {
- node = node->next;
- if (node == list)
- {
- node = (rt_list_t *)RT_NULL;
- break;
- }
- nr++;
- *array++ = node;
- if (nr == arg->nr)
- {
- break;
- }
- }
- rt_spin_unlock_irqrestore(&info->spinlock, level);
- arg->nr_out = nr;
- return node;
- }
- static char *const device_type_str[RT_Device_Class_Unknown] =
- {
- "Character Device",
- "Block Device",
- "Network Interface",
- "MTD Device",
- "CAN Device",
- "RTC",
- "Sound Device",
- "Graphic Device",
- "I2C Bus",
- "USB Slave Device",
- "USB Host Bus",
- "USB OTG Bus",
- "SPI Bus",
- "SPI Device",
- "SDIO Bus",
- "PM Pseudo Device",
- "Pipe",
- "Portal Device",
- "Timer Device",
- "Miscellaneous Device",
- "Sensor Device",
- "Touch Device",
- "Phy Device",
- "Security Device",
- "WLAN Device",
- "Pin Device",
- "ADC Device",
- "DAC Device",
- "WDT Device",
- "PWM Device",
- "Bus Device",
- };
- static void save_info(struct device_show *dev, char *dev_name)
- {
- char tmp[256] = {0};
- int len;
- dev->index ++;
- rt_snprintf(tmp, 256, "%d %s\n", dev->index, dev_name);
- tmp[255] = 0;
- len = rt_strlen(tmp);
- if (dev->size > dev->len + len)
- {
- strcat(dev->buf, tmp);
- dev->len += len;
- }
- else
- {
- if (dev->buf == RT_NULL)
- {
- dev->buf = rt_calloc(1, 4096);
- }
- else
- {
- dev->buf = rt_realloc(dev->buf, dev->size + 4096);
- }
- if (dev->buf)
- {
- dev->size += 4096;
- strcat(dev->buf, tmp);
- dev->len += len;
- }
- }
- }
- static void list_device(struct device_show *dev)
- {
- rt_base_t level;
- list_get_next_t find_arg;
- struct rt_object_information *info;
- rt_list_t *obj_list[LIST_FIND_OBJ_NR];
- rt_list_t *next = (rt_list_t *)RT_NULL;
- list_find_init(&find_arg, RT_Object_Class_Device, obj_list, sizeof(obj_list) / sizeof(obj_list[0]));
- info = rt_list_entry(find_arg.list, struct rt_object_information, object_list);
- do
- {
- next = list_get_next(next, &find_arg);
- {
- int i;
- for (i = 0; i < find_arg.nr_out; i++)
- {
- struct rt_object *obj;
- struct rt_device *device;
- obj = rt_list_entry(obj_list[i], struct rt_object, list);
- level = rt_spin_lock_irqsave(&info->spinlock);
- if ((obj->type & ~RT_Object_Class_Static) != find_arg.type)
- {
- rt_spin_unlock_irqrestore(&info->spinlock, level);
- continue;
- }
- rt_spin_unlock_irqrestore(&info->spinlock, level);
- device = (struct rt_device *)obj;
- if (device->type < RT_Device_Class_Unknown)
- {
- save_info(dev + device->type, device->parent.name);
- }
- }
- }
- }
- while (next != (rt_list_t *)RT_NULL);
- }
- static int show_info(struct dfs_seq_file *seq)
- {
- struct device_show _show[RT_Device_Class_Unknown] = {0};
- list_device(_show);
- for (int i = 0; i < RT_Device_Class_Unknown; i++)
- {
- if (_show[i].buf)
- {
- dfs_seq_printf(seq, "%s:\n", device_type_str[i]);
- dfs_seq_write(seq, _show[i].buf, _show[i].len);
- dfs_seq_putc(seq, '\n');
- rt_free(_show[i].buf);
- }
- }
- return 0;
- }
- static void *seq_start(struct dfs_seq_file *seq, off_t *index)
- {
- off_t i = *index; // seq->index
- return NULL + (i == 0);
- }
- static void seq_stop(struct dfs_seq_file *seq, void *data)
- {
- }
- static void *seq_next(struct dfs_seq_file *seq, void *data, off_t *index)
- {
- /* data: The return value of the start or next*/
- off_t i = *index + 1; // seq->index
- *index = i;
- return NULL;
- }
- static int seq_show(struct dfs_seq_file *seq, void *data)
- {
- /* data: The return value of the start or next*/
- show_info(seq);
- return 0;
- }
- static const struct dfs_seq_ops seq_ops = {
- .start = seq_start,
- .stop = seq_stop,
- .next = seq_next,
- .show = seq_show,
- };
- int proc_devices_init(void)
- {
- struct proc_dentry *dentry = proc_create_data("devices", 0, NULL, NULL, NULL);
- if (dentry)
- {
- dentry->seq_ops = &seq_ops;
- }
- proc_release(dentry);
- return 0;
- }
- INIT_ENV_EXPORT(proc_devices_init);
|