2 * netifd - network interface daemon
3 * Copyright (C) 2012 Felix Fietkau <nbd@openwrt.org>
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License version 2
7 * as published by the Free Software Foundation
9 * This program is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 * GNU General Public License for more details.
17 #include <arpa/inet.h>
18 #include <netinet/in.h>
19 #include <sys/socket.h>
28 __vlist_simple_init(struct vlist_simple_tree
*tree
, int offset
)
30 INIT_LIST_HEAD(&tree
->list
);
32 tree
->head_offset
= offset
;
36 vlist_simple_delete(struct vlist_simple_tree
*tree
, struct vlist_simple_node
*node
)
40 list_del(&node
->list
);
41 ptr
= (char *) node
- tree
->head_offset
;
46 vlist_simple_flush(struct vlist_simple_tree
*tree
)
48 struct vlist_simple_node
*n
, *tmp
;
50 list_for_each_entry_safe(n
, tmp
, &tree
->list
, list
) {
51 if ((n
->version
== tree
->version
|| n
->version
== -1) &&
55 vlist_simple_delete(tree
, n
);
60 vlist_simple_replace(struct vlist_simple_tree
*dest
, struct vlist_simple_tree
*old
)
62 struct vlist_simple_node
*n
, *tmp
;
64 vlist_simple_update(dest
);
65 list_for_each_entry_safe(n
, tmp
, &old
->list
, list
) {
67 vlist_simple_add(dest
, n
);
69 vlist_simple_flush(dest
);
73 vlist_simple_flush_all(struct vlist_simple_tree
*tree
)
76 vlist_simple_flush(tree
);
80 parse_netmask_string(const char *str
, bool v6
)
86 if (!strchr(str
, '.')) {
87 ret
= strtoul(str
, &err
, 0);
97 if (inet_aton(str
, &addr
) != 1)
100 return 32 - fls(~(ntohl(addr
.s_addr
)));
107 split_netmask(char *str
, unsigned int *netmask
, bool v6
)
109 char *delim
= strchr(str
, '/');
114 *netmask
= parse_netmask_string(delim
, v6
);
120 parse_ip_and_netmask(int af
, const char *str
, void *addr
, unsigned int *netmask
)
122 char *astr
= alloca(strlen(str
) + 1);
125 if (!split_netmask(astr
, netmask
, af
== AF_INET6
))
128 if (af
== AF_INET6
) {
136 return inet_pton(af
, astr
, addr
);
140 format_macaddr(uint8_t *mac
)
142 static char str
[sizeof("ff:ff:ff:ff:ff:ff ")];
144 snprintf(str
, sizeof(str
), "%02x:%02x:%02x:%02x:%02x:%02x",
145 mac
[0], mac
[1], mac
[2], mac
[3], mac
[4], mac
[5]);
153 static uint32_t *crcvals
= NULL
;
155 crcvals
= malloc(sizeof(*crcvals
) * 256);
157 for (size_t i
= 0; i
< 256; ++i
) {
159 for (size_t j
= 0; j
< 8; ++j
)
160 c
= (c
& 1) ? (0xEDB88320 ^ (c
>> 1)) : (c
>> 1);
167 uint32_t c
= 0xFFFFFFFF;
170 len
= fread(buf
, 1, sizeof(buf
), fp
);
171 for (size_t i
= 0; i
< len
; ++i
)
172 c
= crcvals
[(c
^ buf
[i
]) & 0xFF] ^ (c
>> 8);
173 } while (len
== sizeof(buf
));
175 return c
^ 0xFFFFFFFF;
178 bool check_pid_path(int pid
, const char *exe
)
180 const char deleted
[] = " (deleted)";
181 const int deleted_len
= strlen(deleted
);
183 int exe_len
= strlen(exe
);
186 char proc_exe_buf
[PROC_PIDPATHINFO_SIZE
];
188 proc_exe_len
= proc_pidpath(pid
, proc_exe_buf
, sizeof(proc_exe_buf
));
191 char *proc_exe_buf
= alloca(exe_len
);
193 sprintf(proc_exe
, "/proc/%d/exe", pid
);
194 proc_exe_len
= readlink(proc_exe
, proc_exe_buf
, exe_len
);
197 if (proc_exe_len
== exe_len
)
198 return !memcmp(exe
, proc_exe_buf
, exe_len
);
199 else if (proc_exe_len
== exe_len
+ deleted_len
)
200 return !memcmp(exe
, proc_exe_buf
, exe_len
) &&
201 !memcmp(exe
+ exe_len
, deleted
, deleted_len
);
206 static const char * const uci_validate_name
[__BLOBMSG_TYPE_LAST
] = {
207 [BLOBMSG_TYPE_STRING
] = "string",
208 [BLOBMSG_TYPE_ARRAY
] = "list(string)",
209 [BLOBMSG_TYPE_INT32
] = "uinteger",
210 [BLOBMSG_TYPE_BOOL
] = "bool",
214 uci_get_validate_string(const struct uci_blob_param_list
*p
, int i
)
217 return p
->validate
[i
];
219 else if (uci_validate_name
[p
->params
[i
].type
])
220 return uci_validate_name
[p
->params
[i
].type
];
222 return p
->validate
[BLOBMSG_TYPE_STRING
];