3 * Copyright 2009 Roy Marples <roy@marples.name>
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
8 * 1. Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
10 * 2. Redistributions in binary form must reproduce the above copyright
11 * notice, this list of conditions and the following disclaimer in the
12 * documentation and/or other materials provided with the distribution.
14 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
15 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
16 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
17 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
18 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
19 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
20 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
21 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
22 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
23 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
33 #include "libdhcpcd.h"
35 #define HIST_MAX 10 /* Max history per ssid/bssid */
38 dhcpcd_wi_history_clear(DHCPCD_CONNECTION *con)
42 while (con->wi_history) {
43 h = con->wi_history->next;
44 free(con->wi_history);
50 dhcpcd_wi_scans_free(DHCPCD_WI_SCAN *wis)
61 static DHCPCD_WI_SCAN *
62 dhcpcd_scanresult_new(DHCPCD_CONNECTION *con, DBusMessageIter *array)
64 DBusMessageIter dict, entry, var;
70 wis = calloc(1, sizeof(*wis));
72 dhcpcd_error_set(con, NULL, errno);
76 dbus_message_iter_recurse(array, &dict);
78 dbus_message_iter_get_arg_type(&dict) == DBUS_TYPE_DICT_ENTRY;
79 dbus_message_iter_next(&dict))
81 dbus_message_iter_recurse(&dict, &entry);
82 if (!dhcpcd_iter_get(con, &entry, DBUS_TYPE_STRING, &s))
84 if (dbus_message_iter_get_arg_type(&entry) !=
87 dbus_message_iter_recurse(&entry, &var);
88 if (strcmp(s, "BSSID") == 0) {
89 if (!dhcpcd_iter_get(con, &var, DBUS_TYPE_STRING, &s))
91 strlcpy(wis->bssid, s, sizeof(wis->bssid));
92 } else if (strcmp(s, "Frequency") == 0) {
93 if (!dhcpcd_iter_get(con, &var, DBUS_TYPE_INT32, &i32))
96 } else if (strcmp(s, "Quality") == 0) {
97 if (!dhcpcd_iter_get(con, &var, DBUS_TYPE_INT32, &i32))
99 wis->quality.value = i32;
100 } else if (strcmp(s, "Noise") == 0) {
101 if (!dhcpcd_iter_get(con, &var, DBUS_TYPE_INT32, &i32))
103 wis->noise.value = i32;
104 } else if (strcmp(s, "Level") == 0) {
105 if (!dhcpcd_iter_get(con, &var, DBUS_TYPE_INT32, &i32))
107 wis->level.value = i32;
108 } else if (strcmp(s, "Flags") == 0) {
109 if (!dhcpcd_iter_get(con, &var, DBUS_TYPE_STRING, &s))
111 strlcpy(wis->flags, s, sizeof(wis->flags));
112 } else if (strcmp(s, "SSID") == 0) {
113 if (!dhcpcd_iter_get(con, &var, DBUS_TYPE_STRING, &s))
115 strlcpy(wis->ssid, s, sizeof(wis->ssid));
118 if (dbus_message_iter_get_arg_type(&dict) != DBUS_TYPE_INVALID) {
119 if (con->errors == errors)
120 dhcpcd_error_set(con, NULL, EINVAL);
128 dhcpcd_wi_scans(DHCPCD_CONNECTION *con, DHCPCD_IF *i)
131 DBusMessageIter args, array;
132 DHCPCD_WI_SCAN *wis, *scans, *l;
133 DHCPCD_WI_HIST *h, *hl;
136 msg = dhcpcd_message_reply(con, "ScanResults", i->ifname);
137 if (!dbus_message_iter_init(msg, &args) ||
138 dbus_message_iter_get_arg_type(&args) != DBUS_TYPE_ARRAY)
140 dhcpcd_error_set(con, NULL, EINVAL);
145 errors = con->errors;
146 dbus_message_iter_recurse(&args, &array);
148 dbus_message_iter_get_arg_type(&array) == DBUS_TYPE_ARRAY;
149 dbus_message_iter_next(&array))
151 wis = dhcpcd_scanresult_new(con, &array);
156 wis->quality.average = wis->quality.value;
157 wis->noise.average = wis->noise.value;
158 wis->level.average = wis->level.value;
159 for (h = con->wi_history; h; h = h->next) {
160 if (strcmp(h->ifname, i->ifname) == 0 &&
161 strcmp(h->bssid, wis->bssid) == 0)
163 wis->quality.average += h->quality;
164 wis->noise.average += h->noise;
165 wis->level.average += h->level;
166 if (++nh == HIST_MAX) {
175 wis->quality.average /= nh;
176 wis->noise.average /= nh;
177 wis->level.average /= nh;
179 h = malloc(sizeof(*h));
181 strlcpy(h->ifname, i->ifname, sizeof(h->ifname));
182 strlcpy(h->bssid, wis->bssid, sizeof(h->bssid));
183 h->quality = wis->quality.value;
184 h->noise = wis->noise.value;
185 h->level = wis->level.value;
186 h->next = con->wi_history;
196 if (dbus_message_iter_get_arg_type(&array) != DBUS_TYPE_INVALID) {
197 if (con->errors == errors)
198 dhcpcd_error_set(con, NULL, EINVAL);
199 dhcpcd_wi_scans_free(scans);
202 dbus_message_unref(msg);
207 dhcpcd_wpa_find_network(DHCPCD_CONNECTION *con, DHCPCD_IF *i, const char *ssid)
210 DBusMessageIter args, array, entry;
215 msg = dhcpcd_message_reply(con, "ListNetworks", i->ifname);
218 if (!dbus_message_iter_init(msg, &args)) {
219 dhcpcd_error_set(con, NULL, EINVAL);
223 errors = con->errors;
225 dbus_message_iter_get_arg_type(&args) == DBUS_TYPE_ARRAY;
226 dbus_message_iter_next(&args))
228 dbus_message_iter_recurse(&args, &array);
230 dbus_message_iter_get_arg_type(&array) == DBUS_TYPE_STRUCT;
231 dbus_message_iter_next(&array))
233 dbus_message_iter_recurse(&array, &entry);
234 if (!dhcpcd_iter_get(con, &entry,
235 DBUS_TYPE_INT32, &id) ||
236 !dhcpcd_iter_get(con, &entry,
237 DBUS_TYPE_STRING, &s))
239 if (strcmp(s, ssid) == 0) {
240 dbus_message_unref(msg);
245 if (dbus_message_iter_get_arg_type(&args) != DBUS_TYPE_INVALID &&
246 con->errors == errors)
247 dhcpcd_error_set(con, NULL, EINVAL);
248 dbus_message_unref(msg);
253 dhcpcd_wpa_add_network(DHCPCD_CONNECTION *con, DHCPCD_IF *i)
256 DBusMessageIter args;
260 msg = dhcpcd_message_reply(con, "AddNetwork", i->ifname);
264 if (dbus_message_iter_init(msg, &args)) {
265 if (dhcpcd_iter_get(con, &args, DBUS_TYPE_INT32, &id))
268 dhcpcd_error_set(con, NULL, EINVAL);
269 dbus_message_unref(msg);
274 dhcpcd_wpa_set_network(DHCPCD_CONNECTION *con, DHCPCD_IF *i, int id,
275 const char *opt, const char *val)
277 DBusMessage *msg, *reply;
278 DBusMessageIter args;
282 msg = dbus_message_new_method_call(DHCPCD_SERVICE, DHCPCD_PATH,
283 DHCPCD_SERVICE, "SetNetwork");
285 dhcpcd_error_set(con, 0, errno);
288 dbus_message_iter_init_append(msg, &args);
290 dbus_message_iter_append_basic(&args, DBUS_TYPE_STRING, &ifname);
291 dbus_message_iter_append_basic(&args, DBUS_TYPE_INT32, &id);
292 dbus_message_iter_append_basic(&args, DBUS_TYPE_STRING, &opt);
293 dbus_message_iter_append_basic(&args, DBUS_TYPE_STRING, &val);
294 reply = dhcpcd_send_reply(con, msg);
295 dbus_message_unref(msg);
299 dbus_message_unref(reply);
306 dhcpcd_wpa_find_network_new(DHCPCD_CONNECTION *con, DHCPCD_IF *i,
314 len = strlen(ssid) + 3;
317 dhcpcd_error_set(con, 0, errno);
320 errors = con->errors;
321 id = dhcpcd_wpa_find_network(con, i, ssid);
322 if (id != -1 || con->errors != errors) {
326 id = dhcpcd_wpa_add_network(con, i);
331 snprintf(q, len, "\"%s\"", ssid);
332 retval = dhcpcd_wpa_set_network(con, i, id, "ssid", q);
338 dhcpcd_wpa_command(DHCPCD_CONNECTION *con, DHCPCD_IF *i,
339 const char *cmd, int id)
341 DBusMessage *msg, *reply;
342 DBusMessageIter args;
346 msg = dbus_message_new_method_call(DHCPCD_SERVICE, DHCPCD_PATH,
347 DHCPCD_SERVICE, cmd);
349 dhcpcd_error_set(con, 0, errno);
352 dbus_message_iter_init_append(msg, &args);
354 dbus_message_iter_append_basic(&args, DBUS_TYPE_STRING, &ifname);
356 dbus_message_iter_append_basic(&args, DBUS_TYPE_INT32, &id);
357 reply = dhcpcd_send_reply(con, msg);
358 dbus_message_unref(msg);
362 dbus_message_unref(reply);