c2c5fc1eca0baf714be018f7864e2ab1bdc07403
[openwrt/openwrt.git] / package / network / utils / uqmi / files / lib / netifd / proto / qmi.sh
1 #!/bin/sh
2
3 [ -n "$INCLUDE_ONLY" ] || {
4 . /lib/functions.sh
5 . ../netifd-proto.sh
6 init_proto "$@"
7 }
8
9 proto_qmi_init_config() {
10 available=1
11 no_device=1
12 proto_config_add_string "device:device"
13 proto_config_add_string apn
14 proto_config_add_string auth
15 proto_config_add_string username
16 proto_config_add_string password
17 proto_config_add_string pincode
18 proto_config_add_int delay
19 proto_config_add_string modes
20 proto_config_add_string pdptype
21 proto_config_add_int profile
22 proto_config_add_boolean dhcp
23 proto_config_add_boolean dhcpv6
24 proto_config_add_boolean autoconnect
25 proto_config_add_int plmn
26 proto_config_add_int timeout
27 proto_config_add_int mtu
28 proto_config_add_defaults
29 }
30
31 proto_qmi_setup() {
32 local interface="$1"
33 local dataformat connstat plmn_mode mcc mnc
34 local device apn auth username password pincode delay modes pdptype
35 local profile dhcp dhcpv6 autoconnect plmn timeout mtu $PROTO_DEFAULT_OPTIONS
36 local ip4table ip6table
37 local cid_4 pdh_4 cid_6 pdh_6
38 local ip_6 ip_prefix_length gateway_6 dns1_6 dns2_6
39
40 json_get_vars device apn auth username password pincode delay modes
41 json_get_vars pdptype profile dhcp dhcpv6 autoconnect plmn ip4table
42 json_get_vars ip6table timeout mtu $PROTO_DEFAULT_OPTIONS
43
44 [ "$timeout" = "" ] && timeout="10"
45
46 [ "$metric" = "" ] && metric="0"
47
48 [ -n "$ctl_device" ] && device=$ctl_device
49
50 [ -n "$device" ] || {
51 echo "No control device specified"
52 proto_notify_error "$interface" NO_DEVICE
53 proto_set_available "$interface" 0
54 return 1
55 }
56
57 [ -n "$delay" ] && sleep "$delay"
58
59 device="$(readlink -f $device)"
60 [ -c "$device" ] || {
61 echo "The specified control device does not exist"
62 proto_notify_error "$interface" NO_DEVICE
63 proto_set_available "$interface" 0
64 return 1
65 }
66
67 devname="$(basename "$device")"
68 devpath="$(readlink -f /sys/class/usbmisc/$devname/device/)"
69 ifname="$( ls "$devpath"/net )"
70 [ -n "$ifname" ] || {
71 echo "The interface could not be found."
72 proto_notify_error "$interface" NO_IFACE
73 proto_set_available "$interface" 0
74 return 1
75 }
76
77 [ -n "$mtu" ] && {
78 echo "Setting MTU to $mtu"
79 /sbin/ip link set dev $ifname mtu $mtu
80 }
81
82 echo "Waiting for SIM initialization"
83 local uninitialized_timeout=0
84 while uqmi -s -d "$device" --get-pin-status | grep '"UIM uninitialized"' > /dev/null; do
85 [ -e "$device" ] || return 1
86 if [ "$uninitialized_timeout" -lt "$timeout" -o "$timeout" = "0" ]; then
87 let uninitialized_timeout++
88 sleep 1;
89 else
90 echo "SIM not initialized"
91 proto_notify_error "$interface" SIM_NOT_INITIALIZED
92 proto_block_restart "$interface"
93 return 1
94 fi
95 done
96
97 if uqmi -s -d "$device" --uim-get-sim-state | grep -q '"Not supported"\|"Invalid QMI command"' &&
98 uqmi -s -d "$device" --get-pin-status | grep -q '"Not supported"\|"Invalid QMI command"' ; then
99 [ -n "$pincode" ] && {
100 uqmi -s -d "$device" --verify-pin1 "$pincode" > /dev/null || uqmi -s -d "$device" --uim-verify-pin1 "$pincode" > /dev/null || {
101 echo "Unable to verify PIN"
102 proto_notify_error "$interface" PIN_FAILED
103 proto_block_restart "$interface"
104 return 1
105 }
106 }
107 else
108 json_load "$(uqmi -s -d "$device" --get-pin-status)"
109 json_get_var pin1_status pin1_status
110 if [ -z "$pin1_status" ]; then
111 json_load "$(uqmi -s -d "$device" --uim-get-sim-state)"
112 json_get_var pin1_status pin1_status
113 fi
114 json_get_var pin1_verify_tries pin1_verify_tries
115
116 case "$pin1_status" in
117 disabled)
118 echo "PIN verification is disabled"
119 ;;
120 blocked)
121 echo "SIM locked PUK required"
122 proto_notify_error "$interface" PUK_NEEDED
123 proto_block_restart "$interface"
124 return 1
125 ;;
126 not_verified)
127 [ "$pin1_verify_tries" -lt "3" ] && {
128 echo "PIN verify count value is $pin1_verify_tries this is below the limit of 3"
129 proto_notify_error "$interface" PIN_TRIES_BELOW_LIMIT
130 proto_block_restart "$interface"
131 return 1
132 }
133 if [ -n "$pincode" ]; then
134 uqmi -s -d "$device" --verify-pin1 "$pincode" > /dev/null 2>&1 || uqmi -s -d "$device" --uim-verify-pin1 "$pincode" > /dev/null 2>&1 || {
135 echo "Unable to verify PIN"
136 proto_notify_error "$interface" PIN_FAILED
137 proto_block_restart "$interface"
138 return 1
139 }
140 else
141 echo "PIN not specified but required"
142 proto_notify_error "$interface" PIN_NOT_SPECIFIED
143 proto_block_restart "$interface"
144 return 1
145 fi
146 ;;
147 verified)
148 echo "PIN already verified"
149 ;;
150 *)
151 echo "PIN status failed (${pin1_status:-sim_not_present})"
152 proto_notify_error "$interface" PIN_STATUS_FAILED
153 proto_block_restart "$interface"
154 return 1
155 ;;
156 esac
157 json_cleanup
158 fi
159
160 if [ -n "$plmn" ]; then
161 json_load "$(uqmi -s -d "$device" --get-plmn)"
162 json_get_var plmn_mode mode
163 json_get_vars mcc mnc || {
164 mcc=0
165 mnc=0
166 }
167
168 if [ "$plmn" = "0" ]; then
169 if [ "$plmn_mode" != "automatic" ]; then
170 mcc=0
171 mnc=0
172 echo "Setting PLMN to auto"
173 fi
174 elif [ "$mcc" -ne "${plmn:0:3}" -o "$mnc" -ne "${plmn:3}" ]; then
175 mcc=${plmn:0:3}
176 mnc=${plmn:3}
177 echo "Setting PLMN to $plmn"
178 else
179 mcc=""
180 mnc=""
181 fi
182 fi
183
184 if [ -n "$mcc" -a -n "$mnc" ]; then
185 uqmi -s -d "$device" --set-plmn --mcc "$mcc" --mnc "$mnc" > /dev/null 2>&1 || {
186 echo "Unable to set PLMN"
187 proto_notify_error "$interface" PLMN_FAILED
188 proto_block_restart "$interface"
189 return 1
190 }
191 fi
192
193 # Cleanup current state if any
194 uqmi -s -d "$device" --stop-network 0xffffffff --autoconnect > /dev/null 2>&1
195
196 # Go online
197 uqmi -s -d "$device" --set-device-operating-mode online > /dev/null 2>&1
198
199 # Set IP format
200 uqmi -s -d "$device" --set-data-format 802.3 > /dev/null 2>&1
201 uqmi -s -d "$device" --wda-set-data-format 802.3 > /dev/null 2>&1
202 dataformat="$(uqmi -s -d "$device" --wda-get-data-format)"
203
204 if [ "$dataformat" = '"raw-ip"' ]; then
205
206 [ -f /sys/class/net/$ifname/qmi/raw_ip ] || {
207 echo "Device only supports raw-ip mode but is missing this required driver attribute: /sys/class/net/$ifname/qmi/raw_ip"
208 return 1
209 }
210
211 echo "Device does not support 802.3 mode. Informing driver of raw-ip only for $ifname .."
212 echo "Y" > /sys/class/net/$ifname/qmi/raw_ip
213 fi
214
215 uqmi -s -d "$device" --sync > /dev/null 2>&1
216
217 uqmi -s -d "$device" --network-register > /dev/null 2>&1
218
219 echo "Waiting for network registration"
220 sleep 1
221 local registration_timeout=0
222 local registration_state=""
223 while true; do
224 registration_state=$(uqmi -s -d "$device" --get-serving-system 2>/dev/null | jsonfilter -e "@.registration" 2>/dev/null)
225
226 [ "$registration_state" = "registered" ] && break
227
228 if [ "$registration_state" = "searching" ] || [ "$registration_state" = "not_registered" ]; then
229 if [ "$registration_timeout" -lt "$timeout" ] || [ "$timeout" = "0" ]; then
230 [ "$registration_state" = "searching" ] || {
231 echo "Device stopped network registration. Restart network registration"
232 uqmi -s -d "$device" --network-register > /dev/null 2>&1
233 }
234 let registration_timeout++
235 sleep 1
236 continue
237 fi
238 echo "Network registration failed, registration timeout reached"
239 else
240 # registration_state is 'registration_denied' or 'unknown' or ''
241 echo "Network registration failed (reason: '$registration_state')"
242 fi
243
244 proto_notify_error "$interface" NETWORK_REGISTRATION_FAILED
245 proto_block_restart "$interface"
246 return 1
247 done
248
249 [ -n "$modes" ] && uqmi -s -d "$device" --set-network-modes "$modes" > /dev/null 2>&1
250
251 echo "Starting network $interface"
252
253 pdptype="$(echo "$pdptype" | awk '{print tolower($0)}')"
254
255 [ "$pdptype" = "ip" -o "$pdptype" = "ipv6" -o "$pdptype" = "ipv4v6" ] || pdptype="ip"
256
257 if [ "$pdptype" = "ip" ]; then
258 [ -z "$autoconnect" ] && autoconnect=1
259 [ "$autoconnect" = 0 ] && autoconnect=""
260 else
261 [ "$autoconnect" = 1 ] || autoconnect=""
262 fi
263
264 [ "$pdptype" = "ip" -o "$pdptype" = "ipv4v6" ] && {
265 cid_4=$(uqmi -s -d "$device" --get-client-id wds)
266 if ! [ "$cid_4" -eq "$cid_4" ] 2> /dev/null; then
267 echo "Unable to obtain client ID"
268 proto_notify_error "$interface" NO_CID
269 return 1
270 fi
271
272 uqmi -s -d "$device" --set-client-id wds,"$cid_4" --set-ip-family ipv4 > /dev/null 2>&1
273
274 pdh_4=$(uqmi -s -d "$device" --set-client-id wds,"$cid_4" \
275 --start-network \
276 ${apn:+--apn $apn} \
277 ${profile:+--profile $profile} \
278 ${auth:+--auth-type $auth} \
279 ${username:+--username $username} \
280 ${password:+--password $password} \
281 ${autoconnect:+--autoconnect})
282
283 # pdh_4 is a numeric value on success
284 if ! [ "$pdh_4" -eq "$pdh_4" ] 2> /dev/null; then
285 echo "Unable to connect IPv4"
286 uqmi -s -d "$device" --set-client-id wds,"$cid_4" --release-client-id wds > /dev/null 2>&1
287 proto_notify_error "$interface" CALL_FAILED
288 return 1
289 fi
290
291 # Check data connection state
292 connstat=$(uqmi -s -d "$device" --set-client-id wds,"$cid_4" --get-data-status)
293 [ "$connstat" == '"connected"' ] || {
294 echo "No data link!"
295 uqmi -s -d "$device" --set-client-id wds,"$cid_4" --release-client-id wds > /dev/null 2>&1
296 proto_notify_error "$interface" CALL_FAILED
297 return 1
298 }
299 }
300
301 [ "$pdptype" = "ipv6" -o "$pdptype" = "ipv4v6" ] && {
302 cid_6=$(uqmi -s -d "$device" --get-client-id wds)
303 if ! [ "$cid_6" -eq "$cid_6" ] 2> /dev/null; then
304 echo "Unable to obtain client ID"
305 proto_notify_error "$interface" NO_CID
306 return 1
307 fi
308
309 uqmi -s -d "$device" --set-client-id wds,"$cid_6" --set-ip-family ipv6 > /dev/null 2>&1
310
311 pdh_6=$(uqmi -s -d "$device" --set-client-id wds,"$cid_6" \
312 --start-network \
313 ${apn:+--apn $apn} \
314 ${profile:+--profile $profile} \
315 ${auth:+--auth-type $auth} \
316 ${username:+--username $username} \
317 ${password:+--password $password} \
318 ${autoconnect:+--autoconnect})
319
320 # pdh_6 is a numeric value on success
321 if ! [ "$pdh_6" -eq "$pdh_6" ] 2> /dev/null; then
322 echo "Unable to connect IPv6"
323 uqmi -s -d "$device" --set-client-id wds,"$cid_6" --release-client-id wds > /dev/null 2>&1
324 proto_notify_error "$interface" CALL_FAILED
325 return 1
326 fi
327
328 # Check data connection state
329 connstat=$(uqmi -s -d "$device" --set-client-id wds,"$cid_6" --get-data-status)
330 [ "$connstat" == '"connected"' ] || {
331 echo "No data link!"
332 uqmi -s -d "$device" --set-client-id wds,"$cid_6" --release-client-id wds > /dev/null 2>&1
333 proto_notify_error "$interface" CALL_FAILED
334 return 1
335 }
336 }
337
338 echo "Setting up $ifname"
339 proto_init_update "$ifname" 1
340 proto_set_keep 1
341 proto_add_data
342 [ -n "$pdh_4" ] && {
343 json_add_string "cid_4" "$cid_4"
344 json_add_string "pdh_4" "$pdh_4"
345 }
346 [ -n "$pdh_6" ] && {
347 json_add_string "cid_6" "$cid_6"
348 json_add_string "pdh_6" "$pdh_6"
349 }
350 proto_close_data
351 proto_send_update "$interface"
352
353 local zone="$(fw3 -q network "$interface" 2>/dev/null)"
354
355 [ -n "$pdh_6" ] && {
356 if [ -z "$dhcpv6" -o "$dhcpv6" = 0 ]; then
357 json_load "$(uqmi -s -d $device --set-client-id wds,$cid_6 --get-current-settings)"
358 json_select ipv6
359 json_get_var ip_6 ip
360 json_get_var gateway_6 gateway
361 json_get_var dns1_6 dns1
362 json_get_var dns2_6 dns2
363 json_get_var ip_prefix_length ip-prefix-length
364
365 proto_init_update "$ifname" 1
366 proto_set_keep 1
367 proto_add_ipv6_address "$ip_6" "128"
368 proto_add_ipv6_prefix "${ip_6}/${ip_prefix_length}"
369 proto_add_ipv6_route "$gateway_6" "128"
370 [ "$defaultroute" = 0 ] || proto_add_ipv6_route "::0" 0 "$gateway_6" "" "" "${ip_6}/${ip_prefix_length}"
371 [ "$peerdns" = 0 ] || {
372 proto_add_dns_server "$dns1_6"
373 proto_add_dns_server "$dns2_6"
374 }
375 [ -n "$zone" ] && {
376 proto_add_data
377 json_add_string zone "$zone"
378 proto_close_data
379 }
380 proto_send_update "$interface"
381 else
382 json_init
383 json_add_string name "${interface}_6"
384 json_add_string ifname "@$interface"
385 json_add_string proto "dhcpv6"
386 [ -n "$ip6table" ] && json_add_string ip6table "$ip6table"
387 proto_add_dynamic_defaults
388 # RFC 7278: Extend an IPv6 /64 Prefix to LAN
389 json_add_string extendprefix 1
390 [ -n "$zone" ] && json_add_string zone "$zone"
391 json_close_object
392 ubus call network add_dynamic "$(json_dump)"
393 fi
394 }
395
396 [ -n "$pdh_4" ] && {
397 if [ "$dhcp" = 0 ]; then
398 json_load "$(uqmi -s -d $device --set-client-id wds,$cid_4 --get-current-settings)"
399 json_select ipv4
400 json_get_var ip_4 ip
401 json_get_var gateway_4 gateway
402 json_get_var dns1_4 dns1
403 json_get_var dns2_4 dns2
404 json_get_var subnet_4 subnet
405
406 proto_init_update "$ifname" 1
407 proto_set_keep 1
408 proto_add_ipv4_address "$ip_4" "$subnet_4"
409 proto_add_ipv4_route "$gateway_4" "128"
410 [ "$defaultroute" = 0 ] || proto_add_ipv4_route "0.0.0.0" 0 "$gateway_4"
411 [ "$peerdns" = 0 ] || {
412 proto_add_dns_server "$dns1_4"
413 proto_add_dns_server "$dns2_4"
414 }
415 [ -n "$zone" ] && {
416 proto_add_data
417 json_add_string zone "$zone"
418 proto_close_data
419 }
420 proto_send_update "$interface"
421 else
422 json_init
423 json_add_string name "${interface}_4"
424 json_add_string ifname "@$interface"
425 json_add_string proto "dhcp"
426 [ -n "$ip4table" ] && json_add_string ip4table "$ip4table"
427 proto_add_dynamic_defaults
428 [ -n "$zone" ] && json_add_string zone "$zone"
429 json_close_object
430 ubus call network add_dynamic "$(json_dump)"
431 fi
432 }
433 }
434
435 qmi_wds_stop() {
436 local cid="$1"
437 local pdh="$2"
438
439 [ -n "$cid" ] || return
440
441 uqmi -s -d "$device" --set-client-id wds,"$cid" \
442 --stop-network 0xffffffff \
443 --autoconnect > /dev/null 2>&1
444
445 [ -n "$pdh" ] && {
446 uqmi -s -d "$device" --set-client-id wds,"$cid" \
447 --stop-network "$pdh" > /dev/null 2>&1
448 }
449
450 uqmi -s -d "$device" --set-client-id wds,"$cid" \
451 --release-client-id wds > /dev/null 2>&1
452 }
453
454 proto_qmi_teardown() {
455 local interface="$1"
456
457 local device cid_4 pdh_4 cid_6 pdh_6
458 json_get_vars device
459
460 [ -n "$ctl_device" ] && device=$ctl_device
461
462 echo "Stopping network $interface"
463
464 json_load "$(ubus call network.interface.$interface status)"
465 json_select data
466 json_get_vars cid_4 pdh_4 cid_6 pdh_6
467
468 qmi_wds_stop "$cid_4" "$pdh_4"
469 qmi_wds_stop "$cid_6" "$pdh_6"
470
471 proto_init_update "*" 0
472 proto_send_update "$interface"
473 }
474
475 [ -n "$INCLUDE_ONLY" ] || {
476 add_protocol qmi
477 }