wireless: fix cancelling setup
[project/netifd.git] / proto-shell.c
1 /*
2  * netifd - network interface daemon
3  * Copyright (C) 2012 Felix Fietkau <nbd@openwrt.org>
4  *
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
8  *
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.
13  */
14 #define _GNU_SOURCE
15
16 #include <string.h>
17 #include <stdlib.h>
18 #include <stdio.h>
19 #include <signal.h>
20
21 #include <arpa/inet.h>
22 #include <netinet/in.h>
23
24
25 #include "netifd.h"
26 #include "interface.h"
27 #include "interface-ip.h"
28 #include "proto.h"
29 #include "system.h"
30 #include "handler.h"
31
32 static int proto_fd = -1;
33
34 enum proto_shell_sm {
35         S_IDLE,
36         S_SETUP,
37         S_SETUP_ABORT,
38         S_TEARDOWN,
39 };
40
41 struct proto_shell_handler {
42         struct list_head list;
43         struct proto_handler proto;
44         char *config_buf;
45         char *script_name;
46         bool init_available;
47
48         struct uci_blob_param_list config;
49 };
50
51 struct proto_shell_dependency {
52         struct list_head list;
53
54         char *interface;
55         struct proto_shell_state *proto;
56         struct interface_user dep;
57
58         union if_addr host;
59         bool v6;
60 };
61
62 struct proto_shell_state {
63         struct interface_proto_state proto;
64         struct proto_shell_handler *handler;
65         struct blob_attr *config;
66
67         struct uloop_timeout teardown_timeout;
68
69         struct netifd_process script_task;
70         struct netifd_process proto_task;
71
72         enum proto_shell_sm sm;
73         bool proto_task_killed;
74
75         int last_error;
76
77         struct list_head deps;
78 };
79
80 static void
81 proto_shell_check_dependencies(struct proto_shell_state *state)
82 {
83         struct proto_shell_dependency *dep;
84         bool available = true;
85
86         list_for_each_entry(dep, &state->deps, list) {
87                 if (dep->dep.iface)
88                         continue;
89
90                 available = false;
91                 break;
92         }
93
94         interface_set_available(state->proto.iface, available);
95 }
96
97 static void
98 proto_shell_if_up_cb(struct interface_user *dep, struct interface *iface,
99                      enum interface_event ev);
100 static void
101 proto_shell_if_down_cb(struct interface_user *dep, struct interface *iface,
102                        enum interface_event ev);
103
104 static void
105 proto_shell_update_host_dep(struct proto_shell_dependency *dep)
106 {
107         struct interface *iface = NULL;
108
109         if (dep->dep.iface)
110                 goto out;
111
112         if (dep->interface[0])
113                 iface = vlist_find(&interfaces, dep->interface, iface, node);
114
115         iface = interface_ip_add_target_route(&dep->host, dep->v6, iface);
116         if (!iface)
117                 goto out;
118
119         interface_remove_user(&dep->dep);
120         dep->dep.cb = proto_shell_if_down_cb;
121         interface_add_user(&dep->dep, iface);
122
123 out:
124         proto_shell_check_dependencies(dep->proto);
125 }
126
127 static void
128 proto_shell_clear_host_dep(struct proto_shell_state *state)
129 {
130         struct proto_shell_dependency *dep, *tmp;
131
132         list_for_each_entry_safe(dep, tmp, &state->deps, list) {
133                 interface_remove_user(&dep->dep);
134                 list_del(&dep->list);
135                 free(dep);
136         }
137 }
138
139 static int
140 proto_shell_handler(struct interface_proto_state *proto,
141                     enum interface_proto_cmd cmd, bool force)
142 {
143         struct proto_shell_state *state;
144         struct proto_shell_handler *handler;
145         struct netifd_process *proc;
146         static char error_buf[32];
147         const char *argv[7];
148         char *envp[2];
149         const char *action;
150         char *config;
151         int ret, i = 0, j = 0;
152
153         state = container_of(proto, struct proto_shell_state, proto);
154         handler = state->handler;
155         proc = &state->script_task;
156
157         if (cmd == PROTO_CMD_SETUP) {
158                 action = "setup";
159                 state->last_error = -1;
160                 proto_shell_clear_host_dep(state);
161                 state->sm = S_SETUP;
162         } else if (cmd == PROTO_CMD_RENEW) {
163                 if (!(handler->proto.flags & PROTO_FLAG_RENEW_AVAILABLE))
164                         return 0;
165                 action = "renew";
166         } else {
167                 if (state->sm == S_TEARDOWN)
168                         return 0;
169
170                 if (state->script_task.uloop.pending) {
171                         if (state->sm != S_SETUP_ABORT) {
172                                 uloop_timeout_set(&state->teardown_timeout, 1000);
173                                 kill(state->script_task.uloop.pid, SIGTERM);
174                                 if (state->proto_task.uloop.pending)
175                                         kill(state->proto_task.uloop.pid, SIGTERM);
176                                 state->sm = S_SETUP_ABORT;
177                         }
178                         return 0;
179                 }
180
181                 action = "teardown";
182                 state->sm = S_TEARDOWN;
183                 if (state->last_error >= 0) {
184                         snprintf(error_buf, sizeof(error_buf), "ERROR=%d", state->last_error);
185                         envp[j++] = error_buf;
186                 }
187                 uloop_timeout_set(&state->teardown_timeout, 5000);
188         }
189
190         config = blobmsg_format_json(state->config, true);
191         if (!config)
192                 return -1;
193
194         argv[i++] = handler->script_name;
195         argv[i++] = handler->proto.name;
196         argv[i++] = action;
197         argv[i++] = proto->iface->name;
198         argv[i++] = config;
199         if (proto->iface->main_dev.dev)
200                 argv[i++] = proto->iface->main_dev.dev->ifname;
201         argv[i] = NULL;
202         envp[j] = NULL;
203
204         ret = netifd_start_process(argv, envp, proc);
205         free(config);
206
207         return ret;
208 }
209
210 static void
211 proto_shell_if_up_cb(struct interface_user *dep, struct interface *iface,
212                      enum interface_event ev)
213 {
214         struct proto_shell_dependency *pdep;
215
216         if (ev != IFEV_UP && ev != IFEV_UPDATE)
217                 return;
218
219         pdep = container_of(dep, struct proto_shell_dependency, dep);
220         proto_shell_update_host_dep(pdep);
221 }
222
223 static void
224 proto_shell_if_down_cb(struct interface_user *dep, struct interface *iface,
225                        enum interface_event ev)
226 {
227         struct proto_shell_dependency *pdep;
228         struct proto_shell_state *state;
229
230         if (ev == IFEV_UP || ev == IFEV_UPDATE)
231                 return;
232
233         pdep = container_of(dep, struct proto_shell_dependency, dep);
234         interface_remove_user(dep);
235         dep->cb = proto_shell_if_up_cb;
236         interface_add_user(dep, NULL);
237
238         state = pdep->proto;
239         if (state->sm == S_IDLE) {
240                 state->proto.proto_event(&state->proto, IFPEV_LINK_LOST);
241                 proto_shell_handler(&state->proto, PROTO_CMD_TEARDOWN, false);
242         }
243 }
244
245 static void
246 proto_shell_task_finish(struct proto_shell_state *state,
247                         struct netifd_process *task)
248 {
249         switch (state->sm) {
250         case S_IDLE:
251                 if (task == &state->proto_task)
252                         state->proto.proto_event(&state->proto, IFPEV_LINK_LOST);
253                 /* fall through */
254         case S_SETUP:
255                 if (task == &state->proto_task)
256                         proto_shell_handler(&state->proto, PROTO_CMD_TEARDOWN,
257                                             false);
258                 break;
259
260         case S_SETUP_ABORT:
261                 if (state->script_task.uloop.pending ||
262                     state->proto_task.uloop.pending)
263                         break;
264
265                 uloop_timeout_cancel(&state->teardown_timeout);
266                 state->sm = S_IDLE;
267                 proto_shell_handler(&state->proto, PROTO_CMD_TEARDOWN, false);
268                 break;
269
270         case S_TEARDOWN:
271                 if (state->script_task.uloop.pending)
272                         break;
273
274                 if (state->proto_task.uloop.pending) {
275                         if (!state->proto_task_killed)
276                                 kill(state->proto_task.uloop.pid, SIGTERM);
277                         break;
278                 }
279
280                 uloop_timeout_cancel(&state->teardown_timeout);
281                 state->sm = S_IDLE;
282                 state->proto.proto_event(&state->proto, IFPEV_DOWN);
283                 break;
284         }
285 }
286
287 static void
288 proto_shell_teardown_timeout_cb(struct uloop_timeout *timeout)
289 {
290         struct proto_shell_state *state;
291
292         state = container_of(timeout, struct proto_shell_state, teardown_timeout);
293
294         netifd_kill_process(&state->script_task);
295         netifd_kill_process(&state->proto_task);
296         proto_shell_task_finish(state, NULL);
297 }
298
299 static void
300 proto_shell_script_cb(struct netifd_process *p, int ret)
301 {
302         struct proto_shell_state *state;
303
304         state = container_of(p, struct proto_shell_state, script_task);
305         proto_shell_task_finish(state, p);
306 }
307
308 static void
309 proto_shell_task_cb(struct netifd_process *p, int ret)
310 {
311         struct proto_shell_state *state;
312
313         state = container_of(p, struct proto_shell_state, proto_task);
314
315         if (state->sm == S_IDLE || state->sm == S_SETUP)
316                 state->last_error = WEXITSTATUS(ret);
317
318         proto_shell_task_finish(state, p);
319 }
320
321 static void
322 proto_shell_free(struct interface_proto_state *proto)
323 {
324         struct proto_shell_state *state;
325
326         state = container_of(proto, struct proto_shell_state, proto);
327         uloop_timeout_cancel(&state->teardown_timeout);
328         proto_shell_clear_host_dep(state);
329         netifd_kill_process(&state->script_task);
330         netifd_kill_process(&state->proto_task);
331         free(state->config);
332         free(state);
333 }
334
335 static void
336 proto_shell_parse_route_list(struct interface *iface, struct blob_attr *attr,
337                              bool v6)
338 {
339         struct blob_attr *cur;
340         int rem;
341
342         blobmsg_for_each_attr(cur, attr, rem) {
343                 if (blobmsg_type(cur) != BLOBMSG_TYPE_TABLE) {
344                         DPRINTF("Ignore wrong route type: %d\n", blobmsg_type(cur));
345                         continue;
346                 }
347
348                 interface_ip_add_route(iface, cur, v6);
349         }
350 }
351
352 static void
353 proto_shell_parse_data(struct interface *iface, struct blob_attr *attr)
354 {
355         struct blob_attr *cur;
356         int rem;
357
358         blobmsg_for_each_attr(cur, attr, rem)
359                 interface_add_data(iface, cur);
360 }
361
362 static struct device *
363 proto_shell_create_tunnel(const char *name, struct blob_attr *attr)
364 {
365         struct device *dev;
366         struct blob_buf b;
367
368         memset(&b, 0, sizeof(b));
369         blob_buf_init(&b, 0);
370         blob_put(&b, 0, blobmsg_data(attr), blobmsg_data_len(attr));
371         dev = device_create(name, &tunnel_device_type, blob_data(b.head));
372         blob_buf_free(&b);
373
374         return dev;
375 }
376
377 enum {
378         NOTIFY_ACTION,
379         NOTIFY_ERROR,
380         NOTIFY_COMMAND,
381         NOTIFY_ENV,
382         NOTIFY_SIGNAL,
383         NOTIFY_AVAILABLE,
384         NOTIFY_LINK_UP,
385         NOTIFY_IFNAME,
386         NOTIFY_ADDR_EXT,
387         NOTIFY_ROUTES,
388         NOTIFY_ROUTES6,
389         NOTIFY_TUNNEL,
390         NOTIFY_DATA,
391         NOTIFY_KEEP,
392         NOTIFY_HOST,
393         NOTIFY_DNS,
394         NOTIFY_DNS_SEARCH,
395         __NOTIFY_LAST
396 };
397
398 static const struct blobmsg_policy notify_attr[__NOTIFY_LAST] = {
399         [NOTIFY_ACTION] = { .name = "action", .type = BLOBMSG_TYPE_INT32 },
400         [NOTIFY_ERROR] = { .name = "error", .type = BLOBMSG_TYPE_ARRAY },
401         [NOTIFY_COMMAND] = { .name = "command", .type = BLOBMSG_TYPE_ARRAY },
402         [NOTIFY_ENV] = { .name = "env", .type = BLOBMSG_TYPE_ARRAY },
403         [NOTIFY_SIGNAL] = { .name = "signal", .type = BLOBMSG_TYPE_INT32 },
404         [NOTIFY_AVAILABLE] = { .name = "available", .type = BLOBMSG_TYPE_BOOL },
405         [NOTIFY_LINK_UP] = { .name = "link-up", .type = BLOBMSG_TYPE_BOOL },
406         [NOTIFY_IFNAME] = { .name = "ifname", .type = BLOBMSG_TYPE_STRING },
407         [NOTIFY_ADDR_EXT] = { .name = "address-external", .type = BLOBMSG_TYPE_BOOL },
408         [NOTIFY_ROUTES] = { .name = "routes", .type = BLOBMSG_TYPE_ARRAY },
409         [NOTIFY_ROUTES6] = { .name = "routes6", .type = BLOBMSG_TYPE_ARRAY },
410         [NOTIFY_TUNNEL] = { .name = "tunnel", .type = BLOBMSG_TYPE_TABLE },
411         [NOTIFY_DATA] = { .name = "data", .type = BLOBMSG_TYPE_TABLE },
412         [NOTIFY_KEEP] = { .name = "keep", .type = BLOBMSG_TYPE_BOOL },
413         [NOTIFY_HOST] = { .name = "host", .type = BLOBMSG_TYPE_STRING },
414         [NOTIFY_DNS] = { .name = "dns", .type = BLOBMSG_TYPE_ARRAY },
415         [NOTIFY_DNS_SEARCH] = { .name = "dns_search", .type = BLOBMSG_TYPE_ARRAY },
416 };
417
418 static int
419 proto_shell_update_link(struct proto_shell_state *state, struct blob_attr *data, struct blob_attr **tb)
420 {
421         struct interface *iface = state->proto.iface;
422         struct blob_attr *cur;
423         struct device *dev;
424         const char *devname;
425         int dev_create = 1;
426         bool addr_ext = false;
427         bool keep = false;
428         bool up;
429
430         if (!tb[NOTIFY_LINK_UP])
431                 return UBUS_STATUS_INVALID_ARGUMENT;
432
433         up = blobmsg_get_bool(tb[NOTIFY_LINK_UP]);
434         if (!up) {
435                 state->proto.proto_event(&state->proto, IFPEV_LINK_LOST);
436                 return 0;
437         }
438
439         if ((cur = tb[NOTIFY_KEEP]) != NULL)
440                 keep = blobmsg_get_bool(cur);
441
442         if ((cur = tb[NOTIFY_ADDR_EXT]) != NULL) {
443                 addr_ext = blobmsg_get_bool(cur);
444                 if (addr_ext)
445                         dev_create = 2;
446         }
447
448         if (iface->state != IFS_UP || !iface->l3_dev.dev)
449                 keep = false;
450
451         if (!keep) {
452                 dev = iface->main_dev.dev;
453                 if (tb[NOTIFY_IFNAME]) {
454                         keep = false;
455                         devname = blobmsg_data(tb[NOTIFY_IFNAME]);
456                         if (tb[NOTIFY_TUNNEL])
457                                 dev = proto_shell_create_tunnel(devname, tb[NOTIFY_TUNNEL]);
458                         else
459                                 dev = device_get(devname, dev_create);
460                 }
461
462                 if (!dev)
463                         return UBUS_STATUS_INVALID_ARGUMENT;
464
465                 interface_set_l3_dev(iface, dev);
466                 device_claim(&iface->l3_dev);
467                 device_set_present(dev, true);
468
469                 interface_update_start(iface);
470         }
471
472         proto_apply_ip_settings(iface, data, addr_ext);
473
474         if ((cur = tb[NOTIFY_ROUTES]) != NULL)
475                 proto_shell_parse_route_list(state->proto.iface, cur, false);
476
477         if ((cur = tb[NOTIFY_ROUTES6]) != NULL)
478                 proto_shell_parse_route_list(state->proto.iface, cur, true);
479
480         if ((cur = tb[NOTIFY_DNS]))
481                 interface_add_dns_server_list(&iface->proto_ip, cur);
482
483         if ((cur = tb[NOTIFY_DNS_SEARCH]))
484                 interface_add_dns_search_list(&iface->proto_ip, cur);
485
486         if ((cur = tb[NOTIFY_DATA]))
487                 proto_shell_parse_data(state->proto.iface, cur);
488
489         interface_update_complete(state->proto.iface);
490
491         if (!keep)
492                 state->proto.proto_event(&state->proto, IFPEV_UP);
493         state->sm = S_IDLE;
494
495         return 0;
496 }
497
498 static bool
499 fill_string_list(struct blob_attr *attr, char **argv, int max)
500 {
501         struct blob_attr *cur;
502         int argc = 0;
503         int rem;
504
505         if (!attr)
506                 goto out;
507
508         blobmsg_for_each_attr(cur, attr, rem) {
509                 if (blobmsg_type(cur) != BLOBMSG_TYPE_STRING)
510                         return false;
511
512                 if (!blobmsg_check_attr(cur, NULL))
513                         return false;
514
515                 argv[argc++] = blobmsg_data(cur);
516                 if (argc == max - 1)
517                         return false;
518         }
519
520 out:
521         argv[argc] = NULL;
522         return true;
523 }
524
525 static int
526 proto_shell_run_command(struct proto_shell_state *state, struct blob_attr **tb)
527 {
528         static char *argv[64];
529         static char *env[32];
530
531         if (!tb[NOTIFY_COMMAND])
532                 goto error;
533
534         if (!fill_string_list(tb[NOTIFY_COMMAND], argv, ARRAY_SIZE(argv)))
535                 goto error;
536
537         if (!fill_string_list(tb[NOTIFY_ENV], env, ARRAY_SIZE(env)))
538                 goto error;
539
540         netifd_start_process((const char **) argv, (char **) env, &state->proto_task);
541
542         return 0;
543
544 error:
545         return UBUS_STATUS_INVALID_ARGUMENT;
546 }
547
548 static int
549 proto_shell_kill_command(struct proto_shell_state *state, struct blob_attr **tb)
550 {
551         unsigned int signal = ~0;
552
553         if (tb[NOTIFY_SIGNAL])
554                 signal = blobmsg_get_u32(tb[NOTIFY_SIGNAL]);
555
556         if (signal > 31)
557                 signal = SIGTERM;
558
559         if (state->proto_task.uloop.pending) {
560                 state->proto_task_killed = true;
561                 kill(state->proto_task.uloop.pid, signal);
562         }
563
564         return 0;
565 }
566
567 static int
568 proto_shell_notify_error(struct proto_shell_state *state, struct blob_attr **tb)
569 {
570         struct blob_attr *cur;
571         char *data[16];
572         int n_data = 0;
573         int rem;
574
575         if (!tb[NOTIFY_ERROR])
576                 return UBUS_STATUS_INVALID_ARGUMENT;
577
578         blobmsg_for_each_attr(cur, tb[NOTIFY_ERROR], rem) {
579                 if (n_data + 1 == ARRAY_SIZE(data))
580                         goto error;
581
582                 if (blobmsg_type(cur) != BLOBMSG_TYPE_STRING)
583                         goto error;
584
585                 if (!blobmsg_check_attr(cur, NULL))
586                         goto error;
587
588                 data[n_data++] = blobmsg_data(cur);
589         }
590
591         if (!n_data)
592                 goto error;
593
594         interface_add_error(state->proto.iface, state->handler->proto.name,
595                         data[0], (const char **) &data[1], n_data - 1);
596
597         return 0;
598
599 error:
600         return UBUS_STATUS_INVALID_ARGUMENT;
601 }
602
603 static int
604 proto_shell_block_restart(struct proto_shell_state *state, struct blob_attr **tb)
605 {
606         state->proto.iface->autostart = false;
607         return 0;
608 }
609
610 static int
611 proto_shell_set_available(struct proto_shell_state *state, struct blob_attr **tb)
612 {
613         if (!tb[NOTIFY_AVAILABLE])
614                 return UBUS_STATUS_INVALID_ARGUMENT;
615
616         interface_set_available(state->proto.iface, blobmsg_get_bool(tb[NOTIFY_AVAILABLE]));
617         return 0;
618 }
619
620 static int
621 proto_shell_add_host_dependency(struct proto_shell_state *state, struct blob_attr **tb)
622 {
623         struct proto_shell_dependency *dep;
624         struct blob_attr *host = tb[NOTIFY_HOST];
625         struct blob_attr *ifname_a = tb[NOTIFY_IFNAME];
626         const char *ifname_str = ifname_a ? blobmsg_data(ifname_a) : "";
627         char *ifname;
628
629         if (!host)
630                 return UBUS_STATUS_INVALID_ARGUMENT;
631
632         dep = calloc_a(sizeof(*dep), &ifname, strlen(ifname_str) + 1);
633         if (inet_pton(AF_INET, blobmsg_data(host), &dep->host) < 1) {
634                 if (inet_pton(AF_INET6, blobmsg_data(host), &dep->host) < 1) {
635                         free(dep);
636                         return UBUS_STATUS_INVALID_ARGUMENT;
637                 } else {
638                         dep->v6 = true;
639                 }
640         }
641
642         dep->proto = state;
643         dep->interface = strcpy(ifname, ifname_str);
644
645         dep->dep.cb = proto_shell_if_up_cb;
646         interface_add_user(&dep->dep, NULL);
647         list_add(&dep->list, &state->deps);
648         proto_shell_update_host_dep(dep);
649         if (!dep->dep.iface)
650                 return UBUS_STATUS_NOT_FOUND;
651
652         return 0;
653 }
654
655 static int
656 proto_shell_setup_failed(struct proto_shell_state *state)
657 {
658         switch (state->sm) {
659         case S_IDLE:
660                 state->proto.proto_event(&state->proto, IFPEV_LINK_LOST);
661                 /* fall through */
662         case S_SETUP:
663                 proto_shell_handler(&state->proto, PROTO_CMD_TEARDOWN, false);
664                 break;
665         default:
666                 break;
667         }
668         return 0;
669 }
670
671 static int
672 proto_shell_notify(struct interface_proto_state *proto, struct blob_attr *attr)
673 {
674         struct proto_shell_state *state;
675         struct blob_attr *tb[__NOTIFY_LAST];
676
677         state = container_of(proto, struct proto_shell_state, proto);
678
679         blobmsg_parse(notify_attr, __NOTIFY_LAST, tb, blob_data(attr), blob_len(attr));
680         if (!tb[NOTIFY_ACTION])
681                 return UBUS_STATUS_INVALID_ARGUMENT;
682
683         switch(blobmsg_get_u32(tb[NOTIFY_ACTION])) {
684         case 0:
685                 return proto_shell_update_link(state, attr, tb);
686         case 1:
687                 return proto_shell_run_command(state, tb);
688         case 2:
689                 return proto_shell_kill_command(state, tb);
690         case 3:
691                 return proto_shell_notify_error(state, tb);
692         case 4:
693                 return proto_shell_block_restart(state, tb);
694         case 5:
695                 return proto_shell_set_available(state, tb);
696         case 6:
697                 return proto_shell_add_host_dependency(state, tb);
698         case 7:
699                 return proto_shell_setup_failed(state);
700         default:
701                 return UBUS_STATUS_INVALID_ARGUMENT;
702         }
703 }
704
705 static struct interface_proto_state *
706 proto_shell_attach(const struct proto_handler *h, struct interface *iface,
707                    struct blob_attr *attr)
708 {
709         struct proto_shell_state *state;
710
711         state = calloc(1, sizeof(*state));
712         INIT_LIST_HEAD(&state->deps);
713
714         state->config = malloc(blob_pad_len(attr));
715         if (!state->config)
716                 goto error;
717
718         memcpy(state->config, attr, blob_pad_len(attr));
719         state->proto.free = proto_shell_free;
720         state->proto.notify = proto_shell_notify;
721         state->proto.cb = proto_shell_handler;
722         state->teardown_timeout.cb = proto_shell_teardown_timeout_cb;
723         state->script_task.cb = proto_shell_script_cb;
724         state->script_task.dir_fd = proto_fd;
725         state->script_task.log_prefix = iface->name;
726         state->proto_task.cb = proto_shell_task_cb;
727         state->proto_task.dir_fd = proto_fd;
728         state->proto_task.log_prefix = iface->name;
729         state->handler = container_of(h, struct proto_shell_handler, proto);
730
731         return &state->proto;
732
733 error:
734         free(state);
735         return NULL;
736 }
737
738 static void
739 proto_shell_add_handler(const char *script, const char *name, json_object *obj)
740 {
741         struct proto_shell_handler *handler;
742         struct proto_handler *proto;
743         json_object *config, *tmp;
744         char *proto_name, *script_name;
745
746         handler = calloc_a(sizeof(*handler),
747                            &proto_name, strlen(name) + 1,
748                            &script_name, strlen(script) + 1);
749         if (!handler)
750                 return;
751
752         handler->script_name = strcpy(script_name, script);
753
754         proto = &handler->proto;
755         proto->name = strcpy(proto_name, name);
756         proto->config_params = &handler->config;
757         proto->attach = proto_shell_attach;
758
759         tmp = json_get_field(obj, "no-device", json_type_boolean);
760         if (tmp && json_object_get_boolean(tmp))
761                 handler->proto.flags |= PROTO_FLAG_NODEV;
762
763         tmp = json_get_field(obj, "available", json_type_boolean);
764         if (tmp && json_object_get_boolean(tmp))
765                 handler->proto.flags |= PROTO_FLAG_INIT_AVAILABLE;
766
767         tmp = json_get_field(obj, "renew-handler", json_type_boolean);
768         if (tmp && json_object_get_boolean(tmp))
769                 handler->proto.flags |= PROTO_FLAG_RENEW_AVAILABLE;
770
771         config = json_get_field(obj, "config", json_type_array);
772         if (config)
773                 handler->config_buf = netifd_handler_parse_config(&handler->config, config);
774
775         DPRINTF("Add handler for script %s: %s\n", script, proto->name);
776         add_proto_handler(proto);
777 }
778
779 void proto_shell_init(void)
780 {
781         proto_fd = netifd_open_subdir("proto");
782         if (proto_fd < 0)
783                 return;
784
785         netifd_init_script_handlers(proto_fd, proto_shell_add_handler);
786 }