blob: 7b5a124fe737433f9c98fdbc38e14791bcbbb99f (
plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
|
#!/bin/sh
[ -n "$INCLUDE_ONLY" ] || {
. /lib/functions.sh
. ../netifd-proto.sh
init_proto "$@"
}
_qdebug() {
logger -t qmi "$@"
}
PROTO_OPTIONS="apn username password pin roaming"
proto_qmid_init_config() {
available=1
no_device=1
proto_config_add_string "device:device"
proto_config_add_string apn
proto_config_add_string username
proto_config_add_string password
proto_config_add_string pin
proto_config_add_string roaming
proto_config_add_defaults
}
# convert /dev/cdc-wdm0 -> modem0
_qmid_convert_devtoname() {
local device="$1"
if echo "${device}" | grep -q "/dev/cdc-wdm"; then
echo "${device/\/dev\/cdc-wdm/modem}"
else
false
fi
}
# check if uqmid already knows the device
_qmi_device_present() {
ubus list "uqmid.modem.$1" 2>/dev/null >/dev/null
}
_qmi_ensure_device_present() {
local name="$1"
local device="$2"
if _qmi_device_present "$name"; then
return 0
fi
ubus call uqmid add_modem "{'name':'$name','device':'$device'}"
_qmi_device_present "$name"
}
_qmi_poll_state() {
local name=$1
local timeout=$2
# in theory we should have multiple timeouts
# we should only have a long timeout if we reach netsearch
for _i in $(seq 1 "$timeout"); do
json_init
json_load "$(ubus call "uqmid.modem.$name" dump)"
json_get_var state_name state_name
case "$state_name" in
FAILED)
false
return
;;
LIVE)
true
return
;;
*)
# unknown state
sleep 1
;;
esac
done
}
_qmi_poll_sim_state() {
local name=$1
local timeout=$2
for _i in $(seq 1 "$timeout"); do
json_init
json_load "$(ubus call "uqmid.modem.$name" dump)"
json_get_var sim_state_name sim_state_name
case "$sim_state_name" in
IDLE|"WAIT UIM PRESENT"|GET_INFO)
# normal wait
sleep 1
;;
READY)
true
return
;;
CHV_PIN|CHV_PUK|FAILED)
false
return
;;
*)
# unknown state
sleep 1
;;
esac
done
}
proto_qmid_setup() {
local interface="$1"
local device apn
local $PROTO_DEFAULT_OPTIONS
local ip4table ip6table
local ip_6 ip_prefix_length gateway_6 dns1_6 dns2_6
json_get_vars device $PROTO_OPTIONS
json_get_vars ip4table ip6table $PROTO_DEFAULT_OPTIONS
[ "$timeout" = "" ] && timeout="10"
[ "$metric" = "" ] && metric="0"
[ -n "$ctl_device" ] && device=$ctl_device
_qdebug "Setting up interface $interface"
[ -n "$device" ] || {
_qdebug "No control device specified"
proto_notify_error "$interface" NO_DEVICE
proto_set_available "$interface"
return 1
}
[ -n "$delay" ] && sleep "$delay"
device="$(readlink -f "$device")"
[ -c "$device" ] || {
_qdebug "The specified control device does not exist"
proto_notify_error "$interface" NO_DEVICE
return 1
}
devname="$(basename "$device")"
devpath="$(readlink -f "/sys/class/usbmisc/$devname/device/")"
ifname="$( ls "$devpath"/net )"
[ -n "$ifname" ] || {
_qdebug "The interface could not be found."
proto_notify_error "$interface" NO_IFACE
return 1
}
# check if uqmi already knows the device
[ -z "$name" ] && name=$(_qmid_convert_devtoname "$device")
if [ -z "$name" ]; then
_qdebug "Name not set and can't derived from device $device."
proto_notify_error "$interface" NO_NAME
return 1
fi
if ! _qmi_ensure_device_present "$name" "$device"; then
# can't create a device
_qdebug "Can't ensure the device"
proto_notify_error "$interface" NO_IFACE_CREATABLE
return 1
fi
# pass configuration to it
ubus call "uqmid.modem.$name" configure "{'apn':'$apn', 'username': '$username', 'password': '$password', 'pin': '$pin', 'roaming':'$roaming'}"
_qmi_poll_sim_state "$name" 30
# check if simcard is fine
json_init
json_load "$(ubus call "uqmid.modem.$name" dump)"
json_get_var state_name state_name
json_get_var sim_state_name sim_state_name
# use simstate as human readable to have more stable "api"
case "$sim_state_name" in
IDLE | "WAIT UIM PRESENT" | GET_INFO)
_qdebug "SIM is still not ready after 30 seconds. Failing!"
proto_notify_error "$interface" SIM_NOT_READY
return 1
;;
CHV_PIN)
# TODO add support for pincode/unlock
_qdebug "SIM REQUIRED PIN! Failing!"
proto_notify_error "$interface" SIM_PIN_REQUIRED
proto_block_restart "$interface"
return 1
;;
CHV_PUK)
_qdebug "SIM REQUIRED PUK! Failing!"
proto_notify_error "$interface" SIM_PUK_REQUIRED
proto_block_restart "$interface"
return 1
;;
READY)
# continue to process
_qdebug "SIM is READY"
;;
FAILED)
_qdebug "SIM is in FAILED state. Failing!"
# blocked state
proto_notify_error "$interface" SIM_FAILED
proto_block_restart "$interface"
return 1
;;
*)
_qdebug "UNKNOWN STATE $sim_state_name"
ubus call "uqmid.modem.$name" dump >> /tmp/qmi.log
# unknown sim state
proto_notify_error "$interface" SIM_STATE_UNKNOWN
return 1
;;
esac
# poll here again for a state
# TODO: until it reaches LIVE or the FSM terminates in a failure mode
_qmi_poll_state "$name" 30
json_init
json_load "$(ubus call "uqmid.modem.$name" dump)"
json_get_var state_name state_name
case "$state_name" in
LIVE)
# found our correct state
;;
*)
_qdebug "Modem $interface didn't entered LIVE. Instead modem is in state $state_name"
proto_notify_error "$interface" NOT_READY_YET
return 1
;;
esac
json_get_var ipv4_addr ipv4_addr
json_get_var ipv4_netmask ipv4_netmask
json_get_var ipv4_gateway ipv4_gateway
json_get_var dns1 dns1
json_get_var dns2 dns2
proto_init_update "$ifname" 1
proto_set_keep 1
proto_add_ipv4_address "$ipv4_addr" "$ipv4_netmask"
proto_add_ipv4_route "$ipv4_gateway" "32"
[ "$defaultroute" = 0 ] || proto_add_ipv4_route "0.0.0.0" 0 "$gateway_4"
[ "$peerdns" = 0 ] || {
[ -n "$dns1" ] && proto_add_dns_server "$dns1"
[ -n "$dns2" ] && proto_add_dns_server "$dns2"
}
[ -n "$zone" ] && {
proto_add_data
json_add_string zone "$zone"
proto_close_data
}
proto_send_update "$interface"
## state:
## - last signal strength <- last, dump from internal state, not query it
## - last network state
## -
# TODO: check if SIM is initialized
# echo "SIM not initialized"
# proto_notify_error "$interface" SIM_NOT_INITIALIZED
# proto_block_restart "$interface"
# return 1
# Check if UIM application is stuck in illegal state
# TODO: proto_notify_error "$interface" NETWORK_REGISTRATION_FAILED
## proto_init_update "$ifname" 1
## proto_set_keep 1
## proto_add_data
## [ -n "$pdh_4" ] && {
## json_add_string "cid_4" "$cid_4"
## json_add_string "pdh_4" "$pdh_4"
## }
## [ -n "$pdh_6" ] && {
## json_add_string "cid_6" "$cid_6"
## json_add_string "pdh_6" "$pdh_6"
## }
## proto_close_data
## proto_send_update "$interface"
# if [ -z "$dhcpv6" -o "$dhcpv6" = 0 ]; then
# json_load "$(uqmi -s -d $device -t 1000 --set-client-id wds,$cid_6 --get-current-settings)"
# json_select ipv6
# json_get_var ip_6 ip
# json_get_var gateway_6 gateway
# json_get_var dns1_6 dns1
# json_get_var dns2_6 dns2
# json_get_var ip_prefix_length ip-prefix-length
#
# proto_init_update "$ifname" 1
# proto_set_keep 1
# proto_add_ipv6_address "$ip_6" "128"
# proto_add_ipv6_prefix "${ip_6}/${ip_prefix_length}"
# proto_add_ipv6_route "$gateway_6" "128"
# [ "$defaultroute" = 0 ] || proto_add_ipv6_route "::0" 0 "$gateway_6" "" "" "${ip_6}/${ip_prefix_length}"
# [ "$peerdns" = 0 ] || {
# proto_add_dns_server "$dns1_6"
# proto_add_dns_server "$dns2_6"
# }
# [ -n "$zone" ] && {
# proto_add_data
# json_add_string zone "$zone"
# proto_close_data
# }
# proto_send_update "$interface"
# else
# json_init
# json_add_string name "${interface}_6"
# json_add_string ifname "@$interface"
# [ "$pdptype" = "ipv4v6" ] && json_add_string iface_464xlat "0"
# json_add_string proto "dhcpv6"
# [ -n "$ip6table" ] && json_add_string ip6table "$ip6table"
# proto_add_dynamic_defaults
# # RFC 7278: Extend an IPv6 /64 Prefix to LAN
# json_add_string extendprefix 1
# [ -n "$zone" ] && json_add_string zone "$zone"
# json_close_object
# ubus call network add_dynamic "$(json_dump)"
# fi
}
proto_qmid_teardown() {
local interface="$1"
local device
json_get_vars device
[ -n "$ctl_device" ] && device=$ctl_device
[ -z "$name" ] && name=$(_qmid_convert_devtoname "$device")
echo "Stopping network $interface"
ubus call uqmid remove_modem "{ 'name': '$name' }"
proto_init_update "*" 0
proto_send_update "$interface"
}
[ -n "$INCLUDE_ONLY" ] || {
add_protocol qmid
}
|