1 /*-*- Mode: C; c-basic-offset: 8 -*-*/
9 #include "conf-parser.h"
10 #include "load-fragment.h"
13 static int config_parse_deps(
23 Name *name = userdata;
33 FOREACH_WORD(w, &l, rvalue, state) {
38 if (!(t = strndup(w, l)))
41 r = manager_load_name(name->meta.manager, t, &other);
48 if (!(*set = set_new(trivial_hash_func, trivial_compare_func)))
51 if ((r = set_put(*set, other)) < 0)
58 static int config_parse_names(
68 Name *name = userdata;
78 FOREACH_WORD(w, &l, rvalue, state) {
83 if (!(t = strndup(w, l)))
86 other = manager_get_name(name->meta.manager, t);
92 if (other->meta.load_state != NAME_STUB) {
97 if ((r = name_merge(name, other)) < 0) {
106 if (!(*set = set_new(trivial_hash_func, trivial_compare_func))) {
111 if ((r = set_put(*set, t)) < 0) {
125 static int config_parse_listen(
126 const char *filename,
145 if (!(p = new0(SocketPort, 1)))
148 if (streq(lvalue, "ListenFIFO")) {
149 p->type = SOCKET_FIFO;
151 if (!(p->path = strdup(rvalue))) {
156 p->type = SOCKET_SOCKET;
158 if ((r = socket_address_parse(&p->address, rvalue)) < 0) {
159 log_error("[%s:%u] Failed to parse address value: %s", filename, line, rvalue);
164 if (streq(lvalue, "ListenStream"))
165 p->address.type = SOCK_STREAM;
166 else if (streq(lvalue, "ListenDatagram"))
167 p->address.type = SOCK_DGRAM;
169 assert(streq(lvalue, "ListenSequentialPacket"));
170 p->address.type = SOCK_SEQPACKET;
173 if (socket_address_family(&p->address) != AF_LOCAL && p->address.type == SOCK_SEQPACKET) {
175 return -EPROTONOSUPPORT;
180 LIST_PREPEND(SocketPort, s->ports, p);
185 static int config_parse_bind(
186 const char *filename,
204 if ((r = parse_boolean(rvalue)) < 0) {
205 log_error("[%s:%u] Failed to parse bind IPv6 only value: %s", filename, line, rvalue);
209 s->bind_ipv6_only = r ? SOCKET_ADDRESS_IPV6_ONLY : SOCKET_ADDRESS_BOTH;
214 int name_load_fragment(Name *n) {
216 static const char* const section_table[_NAME_TYPE_MAX] = {
217 [NAME_SERVICE] = "Service",
218 [NAME_TIMER] = "Timer",
219 [NAME_SOCKET] = "Socket",
220 [NAME_MILESTONE] = "Milestone",
221 [NAME_DEVICE] = "Device",
222 [NAME_MOUNT] = "Mount",
223 [NAME_AUTOMOUNT] = "Automount",
224 [NAME_SNAPSHOT] = "Snapshot"
227 const ConfigItem items[] = {
228 { "Names", config_parse_names, &n->meta.names, "Meta" },
229 { "Description", config_parse_string, &n->meta.description, "Meta" },
230 { "Requires", config_parse_deps, n->meta.dependencies+NAME_REQUIRES, "Meta" },
231 { "SoftRequires", config_parse_deps, n->meta.dependencies+NAME_SOFT_REQUIRES, "Meta" },
232 { "Wants", config_parse_deps, n->meta.dependencies+NAME_WANTS, "Meta" },
233 { "Requisite", config_parse_deps, n->meta.dependencies+NAME_REQUISITE, "Meta" },
234 { "SoftRequisite", config_parse_deps, n->meta.dependencies+NAME_SOFT_REQUISITE, "Meta" },
235 { "Conflicts", config_parse_deps, n->meta.dependencies+NAME_CONFLICTS, "Meta" },
236 { "Before", config_parse_deps, n->meta.dependencies+NAME_BEFORE, "Meta" },
237 { "After", config_parse_deps, n->meta.dependencies+NAME_AFTER, "Meta" },
238 { "ListenStream", config_parse_listen, &n->socket, "Socket" },
239 { "ListenDatagram", config_parse_listen, &n->socket, "Socket" },
240 { "ListenSequentialPacket", config_parse_listen, &n->socket, "Socket" },
241 { "ListenFIFO", config_parse_listen, &n->socket, "Socket" },
242 { "BindIPv6Only", config_parse_bind, &n->socket, "Socket" },
243 { "Backlog", config_parse_unsigned, &n->socket.backlog, "Socket" },
244 { NULL, NULL, NULL, NULL }
252 const char *sections[3];
255 assert(n->meta.load_state == NAME_STUB);
257 sections[0] = "Meta";
258 sections[1] = section_table[n->meta.type];
261 SET_FOREACH(t, n->meta.names, state)
262 if ((r = config_parse(t, sections, items, n)) < 0)