xref: /openbmc/linux/tools/lib/api/fd/array.c (revision 59b4412f)
1 // SPDX-License-Identifier: GPL-2.0-only
2 /*
3  * Copyright (C) 2014, Red Hat Inc, Arnaldo Carvalho de Melo <acme@redhat.com>
4  */
5 #include "array.h"
6 #include <errno.h>
7 #include <fcntl.h>
8 #include <poll.h>
9 #include <stdlib.h>
10 #include <unistd.h>
11 
12 void fdarray__init(struct fdarray *fda, int nr_autogrow)
13 {
14 	fda->entries	 = NULL;
15 	fda->priv	 = NULL;
16 	fda->nr		 = fda->nr_alloc = 0;
17 	fda->nr_autogrow = nr_autogrow;
18 }
19 
20 int fdarray__grow(struct fdarray *fda, int nr)
21 {
22 	void *priv;
23 	int nr_alloc = fda->nr_alloc + nr;
24 	size_t psize = sizeof(fda->priv[0]) * nr_alloc;
25 	size_t size  = sizeof(struct pollfd) * nr_alloc;
26 	struct pollfd *entries = realloc(fda->entries, size);
27 
28 	if (entries == NULL)
29 		return -ENOMEM;
30 
31 	priv = realloc(fda->priv, psize);
32 	if (priv == NULL) {
33 		free(entries);
34 		return -ENOMEM;
35 	}
36 
37 	fda->nr_alloc = nr_alloc;
38 	fda->entries  = entries;
39 	fda->priv     = priv;
40 	return 0;
41 }
42 
43 struct fdarray *fdarray__new(int nr_alloc, int nr_autogrow)
44 {
45 	struct fdarray *fda = calloc(1, sizeof(*fda));
46 
47 	if (fda != NULL) {
48 		if (fdarray__grow(fda, nr_alloc)) {
49 			free(fda);
50 			fda = NULL;
51 		} else {
52 			fda->nr_autogrow = nr_autogrow;
53 		}
54 	}
55 
56 	return fda;
57 }
58 
59 void fdarray__exit(struct fdarray *fda)
60 {
61 	free(fda->entries);
62 	free(fda->priv);
63 	fdarray__init(fda, 0);
64 }
65 
66 void fdarray__delete(struct fdarray *fda)
67 {
68 	fdarray__exit(fda);
69 	free(fda);
70 }
71 
72 int fdarray__add(struct fdarray *fda, int fd, short revents)
73 {
74 	int pos = fda->nr;
75 
76 	if (fda->nr == fda->nr_alloc &&
77 	    fdarray__grow(fda, fda->nr_autogrow) < 0)
78 		return -ENOMEM;
79 
80 	fda->entries[fda->nr].fd     = fd;
81 	fda->entries[fda->nr].events = revents;
82 	fda->nr++;
83 	return pos;
84 }
85 
86 int fdarray__filter(struct fdarray *fda, short revents,
87 		    void (*entry_destructor)(struct fdarray *fda, int fd, void *arg),
88 		    void *arg)
89 {
90 	int fd, nr = 0;
91 
92 	if (fda->nr == 0)
93 		return 0;
94 
95 	for (fd = 0; fd < fda->nr; ++fd) {
96 		if (!fda->entries[fd].events)
97 			continue;
98 
99 		if (fda->entries[fd].revents & revents) {
100 			if (entry_destructor)
101 				entry_destructor(fda, fd, arg);
102 
103 			fda->entries[fd].revents = fda->entries[fd].events = 0;
104 			continue;
105 		}
106 
107 		++nr;
108 	}
109 
110 	return nr;
111 }
112 
113 int fdarray__poll(struct fdarray *fda, int timeout)
114 {
115 	return poll(fda->entries, fda->nr, timeout);
116 }
117 
118 int fdarray__fprintf(struct fdarray *fda, FILE *fp)
119 {
120 	int fd, printed = fprintf(fp, "%d [ ", fda->nr);
121 
122 	for (fd = 0; fd < fda->nr; ++fd)
123 		printed += fprintf(fp, "%s%d", fd ? ", " : "", fda->entries[fd].fd);
124 
125 	return printed + fprintf(fp, " ]");
126 }
127