]> arthur.barton.de Git - netdata.git/blob - src/proc_self_mountinfo.c
Merge pull request #337 from ktsaou/master
[netdata.git] / src / proc_self_mountinfo.c
1 #ifdef HAVE_CONFIG_H
2 #include <config.h>
3 #endif
4 #include <stdio.h>
5 #include <stdlib.h>
6 #include <string.h>
7 #include <sys/types.h>
8 #include <sys/stat.h>
9 #include <fcntl.h>
10
11 #include "common.h"
12 #include "log.h"
13 #include "appconfig.h"
14
15 #include "proc_self_mountinfo.h"
16
17 // find the mount info with the given major:minor
18 // in the supplied linked list of mountinfo structures
19 struct mountinfo *mountinfo_find(struct mountinfo *root, unsigned long major, unsigned long minor) {
20         struct mountinfo *mi;
21
22         for(mi = root; mi ; mi = mi->next)
23                 if(mi->major == major && mi->minor == minor)
24                         return mi;
25
26         return NULL;
27 }
28
29 // find the mount info with the given filesystem and mount_source
30 // in the supplied linked list of mountinfo structures
31 struct mountinfo *mountinfo_find_by_filesystem_mount_source(struct mountinfo *root, const char *filesystem, const char *mount_source) {
32         struct mountinfo *mi;
33         uint32_t filesystem_hash = simple_hash(filesystem), mount_source_hash = simple_hash(mount_source);
34
35         for(mi = root; mi ; mi = mi->next)
36                 if(mi->filesystem_hash == filesystem_hash && mi->mount_source_hash == mount_source_hash
37                                 && !strcmp(mi->filesystem, filesystem) && !strcmp(mi->mount_source, mount_source))
38                         return mi;
39
40         return NULL;
41 }
42
43 // free a linked list of mountinfo structures
44 void mountinfo_free(struct mountinfo *mi) {
45         if(unlikely(!mi))
46                 return;
47
48         if(likely(mi->next))
49                 mountinfo_free(mi->next);
50
51         if(mi->root) free(mi->root);
52         if(mi->mount_point) free(mi->mount_point);
53         if(mi->mount_options) free(mi->mount_options);
54
55 /*
56         if(mi->optional_fields_count) {
57                 int i;
58                 for(i = 0; i < mi->optional_fields_count ; i++)
59                         free(*mi->optional_fields[i]);
60         }
61         free(mi->optional_fields);
62 */
63         free(mi->filesystem);
64         free(mi->mount_source);
65         free(mi->super_options);
66         free(mi);
67 }
68
69 // read the whole mountinfo into a linked list
70 struct mountinfo *mountinfo_read() {
71         procfile *ff = NULL;
72
73         char filename[FILENAME_MAX + 1];
74         snprintf(filename, FILENAME_MAX, "%s/proc/self/mountinfo", global_host_prefix);
75         ff = procfile_open(filename, " \t", PROCFILE_FLAG_DEFAULT);
76         if(!ff) {
77                 snprintf(filename, FILENAME_MAX, "%s/proc/1/mountinfo", global_host_prefix);
78                 ff = procfile_open(filename, " \t", PROCFILE_FLAG_DEFAULT);
79                 if(!ff) return NULL;
80         }
81
82         ff = procfile_readall(ff);
83         if(!ff) return NULL;
84
85         struct mountinfo *root = NULL, *last = NULL, *mi = NULL;
86
87         unsigned long l, lines = procfile_lines(ff);
88         for(l = 0; l < lines ;l++) {
89                 if(procfile_linewords(ff, l) < 5)
90                         continue;
91
92                 mi = malloc(sizeof(struct mountinfo));
93                 if(unlikely(!mi)) fatal("Cannot allocate memory for mountinfo");
94
95                 if(unlikely(!root))
96                         root = last = mi;
97                 else
98                         last->next = mi;
99
100                 last = mi;
101                 mi->next = NULL;
102
103                 unsigned long w = 0;
104                 mi->id = strtoul(procfile_lineword(ff, l, w), NULL, 10); w++;
105                 mi->parentid = strtoul(procfile_lineword(ff, l, w), NULL, 10); w++;
106
107                 char *major = procfile_lineword(ff, l, w), *minor; w++;
108                 for(minor = major; *minor && *minor != ':' ;minor++) ;
109                 *minor = '\0';
110                 minor++;
111
112                 mi->major = strtoul(major, NULL, 10);
113                 mi->minor = strtoul(minor, NULL, 10);
114
115                 mi->root = strdup(procfile_lineword(ff, l, w)); w++;
116                 if(unlikely(!mi->root)) fatal("Cannot allocate memory");
117                 mi->root_hash = simple_hash(mi->root);
118
119                 mi->mount_point = strdup(procfile_lineword(ff, l, w)); w++;
120                 if(unlikely(!mi->mount_point)) fatal("Cannot allocate memory");
121                 mi->mount_point_hash = simple_hash(mi->mount_point);
122
123                 mi->mount_options = strdup(procfile_lineword(ff, l, w)); w++;
124                 if(unlikely(!mi->mount_options)) fatal("Cannot allocate memory");
125
126                 // count the optional fields
127 /*
128                 unsigned long wo = w;
129 */
130                 mi->optional_fields_count = 0;
131                 char *s = procfile_lineword(ff, l, w);
132                 while(*s && *s != '-') {
133                         w++;
134                         s = procfile_lineword(ff, l, w);
135                         mi->optional_fields_count++;
136                 }
137
138 /*
139                 if(unlikely(mi->optional_fields_count)) {
140                         // we have some optional fields
141                         // read them into a new array of pointers;
142
143                         mi->optional_fields = malloc(mi->optional_fields_count * sizeof(char *));
144                         if(unlikely(!mi->optional_fields))
145                                 fatal("Cannot allocate memory for %d mountinfo optional fields", mi->optional_fields_count);
146
147                         int i;
148                         for(i = 0; i < mi->optional_fields_count ; i++) {
149                                 *mi->optional_fields[wo] = strdup(procfile_lineword(ff, l, w));
150                                 if(!mi->optional_fields[wo]) fatal("Cannot allocate memory");
151                                 wo++;
152                         }
153                 }
154                 else
155                         mi->optional_fields = NULL;
156 */
157
158                 if(likely(*s == '-')) {
159                         w++;
160
161                         mi->filesystem = strdup(procfile_lineword(ff, l, w)); w++;
162                         if(!mi->filesystem) fatal("Cannot allocate memory");
163                         mi->filesystem_hash = simple_hash(mi->filesystem);
164
165                         mi->mount_source = strdup(procfile_lineword(ff, l, w)); w++;
166                         if(!mi->mount_source) fatal("Cannot allocate memory");
167                         mi->mount_source_hash = simple_hash(mi->mount_source);
168
169                         mi->super_options = strdup(procfile_lineword(ff, l, w)); w++;
170                         if(!mi->super_options) fatal("Cannot allocate memory");
171                 }
172                 else {
173                         mi->filesystem = NULL;
174                         mi->mount_source = NULL;
175                         mi->super_options = NULL;
176                 }
177
178 /*
179                 info("MOUNTINFO: %u %u %u:%u root '%s', mount point '%s', mount options '%s', filesystem '%s', mount source '%s', super options '%s'",
180                      mi->id,
181                      mi->parentid,
182                      mi->major,
183                      mi->minor,
184                      mi->root,
185                      mi->mount_point,
186                      mi->mount_options,
187                      mi->filesystem,
188                      mi->mount_source,
189                      mi->super_options
190                 );
191 */
192         }
193
194         procfile_close(ff);
195         return root;
196 }