1 /*
2 * WPA Supplicant - auto scan
3 * Copyright (c) 2012, Intel Corporation. All rights reserved.
4 *
5 * This software may be distributed under the terms of the BSD license.
6 * See README for more details.
7 */
8
9 #include "includes.h"
10
11 #include "common.h"
12 #include "config.h"
13 #include "wpa_supplicant_i.h"
14 #include "bss.h"
15 #include "scan.h"
16 #include "autoscan.h"
17
18 #ifdef CONFIG_AUTOSCAN_EXPONENTIAL
19 extern const struct autoscan_ops autoscan_exponential_ops;
20 #endif /* CONFIG_AUTOSCAN_EXPONENTIAL */
21
22 #ifdef CONFIG_AUTOSCAN_PERIODIC
23 extern const struct autoscan_ops autoscan_periodic_ops;
24 #endif /* CONFIG_AUTOSCAN_PERIODIC */
25
26 static const struct autoscan_ops * autoscan_modules[] = {
27 #ifdef CONFIG_AUTOSCAN_EXPONENTIAL
28 &autoscan_exponential_ops,
29 #endif /* CONFIG_AUTOSCAN_EXPONENTIAL */
30 #ifdef CONFIG_AUTOSCAN_PERIODIC
31 &autoscan_periodic_ops,
32 #endif /* CONFIG_AUTOSCAN_PERIODIC */
33 NULL
34 };
35
36
request_scan(struct wpa_supplicant * wpa_s)37 static void request_scan(struct wpa_supplicant *wpa_s)
38 {
39 wpa_s->scan_req = MANUAL_SCAN_REQ;
40
41 if (wpa_supplicant_req_sched_scan(wpa_s))
42 wpa_supplicant_req_scan(wpa_s, wpa_s->scan_interval, 0);
43 }
44
45
autoscan_init(struct wpa_supplicant * wpa_s,int req_scan)46 int autoscan_init(struct wpa_supplicant *wpa_s, int req_scan)
47 {
48 const char *name = wpa_s->conf->autoscan;
49 const char *params;
50 size_t nlen;
51 int i;
52 const struct autoscan_ops *ops = NULL;
53
54 if (wpa_s->autoscan && wpa_s->autoscan_priv)
55 return 0;
56
57 if (name == NULL)
58 return 0;
59
60 params = os_strchr(name, ':');
61 if (params == NULL) {
62 params = "";
63 nlen = os_strlen(name);
64 } else {
65 nlen = params - name;
66 params++;
67 }
68
69 for (i = 0; autoscan_modules[i]; i++) {
70 if (os_strncmp(name, autoscan_modules[i]->name, nlen) == 0) {
71 ops = autoscan_modules[i];
72 break;
73 }
74 }
75
76 if (ops == NULL) {
77 wpa_printf(MSG_ERROR, "autoscan: Could not find module "
78 "matching the parameter '%s'", name);
79 return -1;
80 }
81
82 wpa_s->autoscan_params = NULL;
83
84 wpa_s->autoscan_priv = ops->init(wpa_s, params);
85 if (wpa_s->autoscan_priv == NULL)
86 return -1;
87 wpa_s->autoscan = ops;
88
89 wpa_printf(MSG_DEBUG, "autoscan: Initialized module '%s' with "
90 "parameters '%s'", ops->name, params);
91 if (!req_scan)
92 return 0;
93
94 /*
95 * Cancelling existing scan requests, if any.
96 */
97 wpa_supplicant_cancel_sched_scan(wpa_s);
98 wpa_supplicant_cancel_scan(wpa_s);
99
100 /*
101 * Firing first scan, which will lead to call autoscan_notify_scan.
102 */
103 request_scan(wpa_s);
104
105 return 0;
106 }
107
108
autoscan_deinit(struct wpa_supplicant * wpa_s)109 void autoscan_deinit(struct wpa_supplicant *wpa_s)
110 {
111 if (wpa_s->autoscan && wpa_s->autoscan_priv) {
112 wpa_printf(MSG_DEBUG, "autoscan: Deinitializing module '%s'",
113 wpa_s->autoscan->name);
114 wpa_s->autoscan->deinit(wpa_s->autoscan_priv);
115 wpa_s->autoscan = NULL;
116 wpa_s->autoscan_priv = NULL;
117
118 wpa_s->scan_interval = 5;
119 wpa_s->sched_scan_interval = 0;
120 }
121 }
122
123
autoscan_notify_scan(struct wpa_supplicant * wpa_s,struct wpa_scan_results * scan_res)124 int autoscan_notify_scan(struct wpa_supplicant *wpa_s,
125 struct wpa_scan_results *scan_res)
126 {
127 int interval;
128
129 if (wpa_s->autoscan && wpa_s->autoscan_priv) {
130 interval = wpa_s->autoscan->notify_scan(wpa_s->autoscan_priv,
131 scan_res);
132
133 if (interval <= 0)
134 return -1;
135
136 wpa_s->scan_interval = interval;
137 wpa_s->sched_scan_interval = interval;
138
139 request_scan(wpa_s);
140 }
141
142 return 0;
143 }
144