3 [ -n "$INCLUDE_ONLY" ] ||
{
9 proto_qmi_init_config
() {
12 proto_config_add_string
"device:device"
13 proto_config_add_string apn
14 proto_config_add_string v6apn
15 proto_config_add_string auth
16 proto_config_add_string username
17 proto_config_add_string password
18 proto_config_add_string pincode
19 proto_config_add_int delay
20 proto_config_add_string modes
21 proto_config_add_string pdptype
22 proto_config_add_int profile
23 proto_config_add_int v6profile
24 proto_config_add_boolean dhcp
25 proto_config_add_boolean dhcpv6
26 proto_config_add_boolean autoconnect
27 proto_config_add_int plmn
28 proto_config_add_int timeout
29 proto_config_add_int mtu
30 proto_config_add_defaults
35 local dataformat connstat plmn_mode mcc mnc
36 local device apn v6apn auth username password pincode delay modes pdptype
37 local profile v6profile dhcp dhcpv6 autoconnect plmn timeout mtu
$PROTO_DEFAULT_OPTIONS
38 local ip4table ip6table
39 local cid_4 pdh_4 cid_6 pdh_6
40 local ip_6 ip_prefix_length gateway_6 dns1_6 dns2_6
43 json_get_vars device apn v6apn auth username password pincode delay modes
44 json_get_vars pdptype profile v6profile dhcp dhcpv6 autoconnect plmn ip4table
45 json_get_vars ip6table timeout mtu
$PROTO_DEFAULT_OPTIONS
47 [ "$timeout" = "" ] && timeout
="10"
49 [ "$metric" = "" ] && metric
="0"
51 [ -n "$ctl_device" ] && device
=$ctl_device
54 echo "No control device specified"
55 proto_notify_error
"$interface" NO_DEVICE
56 proto_set_available
"$interface" 0
60 [ -n "$delay" ] && sleep "$delay"
62 device
="$(readlink -f $device)"
64 echo "The specified control device does not exist"
65 proto_notify_error
"$interface" NO_DEVICE
66 proto_set_available
"$interface" 0
70 devname
="$(basename "$device")"
71 devpath
="$(readlink -f /sys/class/usbmisc/$devname/device/)"
72 ifname
="$( ls "$devpath"/net )"
74 echo "The interface could not be found."
75 proto_notify_error
"$interface" NO_IFACE
76 proto_set_available
"$interface" 0
81 echo "Setting MTU to $mtu"
82 /sbin
/ip link
set dev
$ifname mtu
$mtu
85 echo "Waiting for SIM initialization"
86 local uninitialized_timeout
=0
87 # timeout 3s for first call to avoid hanging uqmi
88 uqmi
-d "$device" --get-pin-status -t 3000 > /dev
/null
2>&1
89 while uqmi
-s -d "$device" --get-pin-status |
grep '"UIM uninitialized"' > /dev
/null
; do
90 [ -e "$device" ] ||
return 1
91 if [ "$uninitialized_timeout" -lt "$timeout" -o "$timeout" = "0" ]; then
92 let uninitialized_timeout
++
95 echo "SIM not initialized"
96 proto_notify_error
"$interface" SIM_NOT_INITIALIZED
97 proto_block_restart
"$interface"
102 # Check if UIM application is stuck in illegal state
103 local uim_state_timeout
=0
105 json_load
"$(uqmi -s -d "$device" --uim-get-sim-state)"
106 json_get_var card_application_state card_application_state
108 # SIM card is either completely absent or state is labeled as illegal
109 # Try to power-cycle the SIM card to recover from this state
110 if [ -z "$card_application_state" -o "$card_application_state" = "illegal" ]; then
111 echo "SIM in illegal state - Power-cycling SIM"
113 # Try to reset SIM application
114 uqmi
-d "$device" --uim-power-off --uim-slot 1
116 uqmi
-d "$device" --uim-power-on --uim-slot 1
118 if [ "$uim_state_timeout" -lt "$timeout" ] ||
[ "$timeout" = "0" ]; then
119 let uim_state_timeout
++
125 proto_notify_error
"$interface" SIM_ILLEGAL_STATE
126 proto_block_restart
"$interface"
133 if uqmi
-s -d "$device" --uim-get-sim-state |
grep -q '"Not supported"\|"Invalid QMI command"' &&
134 uqmi
-s -d "$device" --get-pin-status |
grep -q '"Not supported"\|"Invalid QMI command"' ; then
135 [ -n "$pincode" ] && {
136 uqmi
-s -d "$device" --verify-pin1 "$pincode" > /dev
/null || uqmi
-s -d "$device" --uim-verify-pin1 "$pincode" > /dev
/null ||
{
137 echo "Unable to verify PIN"
138 proto_notify_error
"$interface" PIN_FAILED
139 proto_block_restart
"$interface"
144 json_load
"$(uqmi -s -d "$device" --get-pin-status)"
145 json_get_var pin1_status pin1_status
146 if [ -z "$pin1_status" ]; then
147 json_load
"$(uqmi -s -d "$device" --uim-get-sim-state)"
148 json_get_var pin1_status pin1_status
150 json_get_var pin1_verify_tries pin1_verify_tries
152 case "$pin1_status" in
154 echo "PIN verification is disabled"
157 echo "SIM locked PUK required"
158 proto_notify_error
"$interface" PUK_NEEDED
159 proto_block_restart
"$interface"
163 [ "$pin1_verify_tries" -lt "3" ] && {
164 echo "PIN verify count value is $pin1_verify_tries this is below the limit of 3"
165 proto_notify_error
"$interface" PIN_TRIES_BELOW_LIMIT
166 proto_block_restart
"$interface"
169 if [ -n "$pincode" ]; then
170 uqmi
-s -d "$device" --verify-pin1 "$pincode" > /dev
/null
2>&1 || uqmi
-s -d "$device" --uim-verify-pin1 "$pincode" > /dev
/null
2>&1 ||
{
171 echo "Unable to verify PIN"
172 proto_notify_error
"$interface" PIN_FAILED
173 proto_block_restart
"$interface"
177 echo "PIN not specified but required"
178 proto_notify_error
"$interface" PIN_NOT_SPECIFIED
179 proto_block_restart
"$interface"
184 echo "PIN already verified"
187 echo "PIN status failed (${pin1_status:-sim_not_present})"
188 proto_notify_error
"$interface" PIN_STATUS_FAILED
189 proto_block_restart
"$interface"
196 if [ -n "$plmn" ]; then
197 json_load
"$(uqmi -s -d "$device" --get-plmn)"
198 json_get_var plmn_mode mode
199 json_get_vars mcc mnc ||
{
204 if [ "$plmn" = "0" ]; then
205 if [ "$plmn_mode" != "automatic" ]; then
208 echo "Setting PLMN to auto"
210 elif [ "$mcc" -ne "${plmn:0:3}" -o "$mnc" -ne "${plmn:3}" ]; then
213 echo "Setting PLMN to $plmn"
220 # Cleanup current state if any
221 uqmi
-s -d "$device" --stop-network 0xffffffff --autoconnect > /dev
/null
2>&1
222 uqmi
-s -d "$device" --set-ip-family ipv6
--stop-network 0xffffffff --autoconnect > /dev
/null
2>&1
225 uqmi
-s -d "$device" --set-device-operating-mode online
> /dev
/null
2>&1
228 uqmi
-s -d "$device" --set-data-format 802.3 > /dev
/null
2>&1
229 uqmi
-s -d "$device" --wda-set-data-format 802.3 > /dev
/null
2>&1
230 dataformat
="$(uqmi -s -d "$device" --wda-get-data-format)"
232 if [ "$dataformat" = '"raw-ip"' ]; then
234 [ -f /sys
/class
/net
/$ifname/qmi
/raw_ip
] ||
{
235 echo "Device only supports raw-ip mode but is missing this required driver attribute: /sys/class/net/$ifname/qmi/raw_ip"
239 echo "Device does not support 802.3 mode. Informing driver of raw-ip only for $ifname .."
240 echo "Y" > /sys
/class
/net
/$ifname/qmi
/raw_ip
243 uqmi
-s -d "$device" --sync > /dev
/null
2>&1
245 uqmi
-s -d "$device" --network-register > /dev
/null
2>&1
247 # PLMN selection must happen after the call to network-register
248 if [ -n "$mcc" -a -n "$mnc" ]; then
249 uqmi
-s -d "$device" --set-plmn --mcc "$mcc" --mnc "$mnc" > /dev
/null
2>&1 ||
{
250 echo "Unable to set PLMN"
251 proto_notify_error
"$interface" PLMN_FAILED
252 proto_block_restart
"$interface"
258 uqmi
-s -d "$device" --set-network-modes "$modes" > /dev
/null
2>&1
260 # Scan network to not rely on registration-timeout after RAT change
261 uqmi
-s -d "$device" --network-scan > /dev
/null
2>&1
264 echo "Waiting for network registration"
266 local registration_timeout
=0
267 local registration_state
=""
269 registration_state
=$
(uqmi
-s -d "$device" --get-serving-system 2>/dev
/null | jsonfilter
-e "@.registration" 2>/dev
/null
)
271 [ "$registration_state" = "registered" ] && break
273 if [ "$registration_state" = "searching" ] ||
[ "$registration_state" = "not_registered" ]; then
274 if [ "$registration_timeout" -lt "$timeout" ] ||
[ "$timeout" = "0" ]; then
275 [ "$registration_state" = "searching" ] ||
{
276 echo "Device stopped network registration. Restart network registration"
277 uqmi
-s -d "$device" --network-register > /dev
/null
2>&1
279 let registration_timeout
++
283 echo "Network registration failed, registration timeout reached"
285 # registration_state is 'registration_denied' or 'unknown' or ''
286 echo "Network registration failed (reason: '$registration_state')"
289 proto_notify_error
"$interface" NETWORK_REGISTRATION_FAILED
294 echo "Starting network $interface"
296 pdptype
="$(echo "$pdptype" | awk '{print tolower($0)}')"
298 [ "$pdptype" = "ip" -o "$pdptype" = "ipv6" -o "$pdptype" = "ipv4v6" ] || pdptype
="ip"
300 # Configure PDP type and APN for profile 1.
301 # In case GGSN rejects IPv4v6 PDP, modem might not be able to
302 # establish a non-LTE data session.
303 profile_pdptype
="$pdptype"
304 [ "$profile_pdptype" = "ip" ] && profile_pdptype
="ipv4"
305 uqmi
-s -d "$device" --modify-profile "3gpp,1" --apn "$apn" --pdp-type "$profile_pdptype" > /dev
/null
2>&1
307 if [ "$pdptype" = "ip" ]; then
308 [ -z "$autoconnect" ] && autoconnect
=1
309 [ "$autoconnect" = 0 ] && autoconnect
=""
311 [ "$autoconnect" = 1 ] || autoconnect
=""
314 [ "$pdptype" = "ip" -o "$pdptype" = "ipv4v6" ] && {
315 cid_4
=$
(uqmi
-s -d "$device" --get-client-id wds
)
316 if ! [ "$cid_4" -eq "$cid_4" ] 2> /dev
/null
; then
317 echo "Unable to obtain client ID"
318 proto_notify_error
"$interface" NO_CID
322 uqmi
-s -d "$device" --set-client-id wds
,"$cid_4" --set-ip-family ipv4
> /dev
/null
2>&1
324 pdh_4
=$
(uqmi
-s -d "$device" --set-client-id wds
,"$cid_4" \
327 ${profile:+--profile $profile} \
328 ${auth:+--auth-type $auth} \
329 ${username:+--username $username} \
330 ${password:+--password $password} \
331 ${autoconnect:+--autoconnect})
333 # pdh_4 is a numeric value on success
334 if ! [ "$pdh_4" -eq "$pdh_4" ] 2> /dev
/null
; then
335 echo "Unable to connect IPv4"
336 uqmi
-s -d "$device" --set-client-id wds
,"$cid_4" --release-client-id wds
> /dev
/null
2>&1
337 proto_notify_error
"$interface" CALL_FAILED
341 # Check data connection state
342 connstat
=$
(uqmi
-s -d "$device" --set-client-id wds
,"$cid_4" --get-data-status)
343 [ "$connstat" == '"connected"' ] ||
{
345 uqmi
-s -d "$device" --set-client-id wds
,"$cid_4" --release-client-id wds
> /dev
/null
2>&1
346 proto_notify_error
"$interface" CALL_FAILED
351 [ "$pdptype" = "ipv6" -o "$pdptype" = "ipv4v6" ] && {
352 cid_6
=$
(uqmi
-s -d "$device" --get-client-id wds
)
353 if ! [ "$cid_6" -eq "$cid_6" ] 2> /dev
/null
; then
354 echo "Unable to obtain client ID"
355 proto_notify_error
"$interface" NO_CID
359 uqmi
-s -d "$device" --set-client-id wds
,"$cid_6" --set-ip-family ipv6
> /dev
/null
2>&1
362 : "${v6profile:=${profile}}"
364 pdh_6
=$
(uqmi
-s -d "$device" --set-client-id wds
,"$cid_6" \
366 ${v6apn:+--apn $v6apn} \
367 ${v6profile:+--profile $v6profile} \
368 ${auth:+--auth-type $auth} \
369 ${username:+--username $username} \
370 ${password:+--password $password} \
371 ${autoconnect:+--autoconnect})
373 # pdh_6 is a numeric value on success
374 if ! [ "$pdh_6" -eq "$pdh_6" ] 2> /dev
/null
; then
375 echo "Unable to connect IPv6"
376 uqmi
-s -d "$device" --set-client-id wds
,"$cid_6" --release-client-id wds
> /dev
/null
2>&1
377 proto_notify_error
"$interface" CALL_FAILED
381 # Check data connection state
382 connstat
=$
(uqmi
-s -d "$device" --set-client-id wds
,"$cid_6" --set-ip-family ipv6
--get-data-status)
383 [ "$connstat" == '"connected"' ] ||
{
385 uqmi
-s -d "$device" --set-client-id wds
,"$cid_6" --release-client-id wds
> /dev
/null
2>&1
386 proto_notify_error
"$interface" CALL_FAILED
391 echo "Setting up $ifname"
392 proto_init_update
"$ifname" 1
396 json_add_string
"cid_4" "$cid_4"
397 json_add_string
"pdh_4" "$pdh_4"
400 json_add_string
"cid_6" "$cid_6"
401 json_add_string
"pdh_6" "$pdh_6"
404 proto_send_update
"$interface"
406 local zone
="$(fw3 -q network "$interface" 2>/dev/null)"
409 if [ -z "$dhcpv6" -o "$dhcpv6" = 0 ]; then
410 json_load
"$(uqmi -s -d $device --set-client-id wds,$cid_6 --get-current-settings)"
413 json_get_var gateway_6 gateway
414 json_get_var dns1_6 dns1
415 json_get_var dns2_6 dns2
416 json_get_var ip_prefix_length ip-prefix-length
418 proto_init_update
"$ifname" 1
420 proto_add_ipv6_address
"$ip_6" "128"
421 proto_add_ipv6_prefix
"${ip_6}/${ip_prefix_length}"
422 proto_add_ipv6_route
"$gateway_6" "128"
423 [ "$defaultroute" = 0 ] || proto_add_ipv6_route
"::0" 0 "$gateway_6" "" "" "${ip_6}/${ip_prefix_length}"
424 [ "$peerdns" = 0 ] ||
{
425 proto_add_dns_server
"$dns1_6"
426 proto_add_dns_server
"$dns2_6"
430 json_add_string zone
"$zone"
433 proto_send_update
"$interface"
436 json_add_string name
"${interface}_6"
437 json_add_string ifname
"@$interface"
438 [ "$pdptype" = "ipv4v6" ] && json_add_string iface_464xlat
"0"
439 json_add_string proto
"dhcpv6"
440 [ -n "$ip6table" ] && json_add_string ip6table
"$ip6table"
441 proto_add_dynamic_defaults
442 # RFC 7278: Extend an IPv6 /64 Prefix to LAN
443 json_add_string extendprefix
1
444 [ -n "$zone" ] && json_add_string zone
"$zone"
446 ubus call network add_dynamic
"$(json_dump)"
451 if [ "$dhcp" = 0 ]; then
452 json_load
"$(uqmi -s -d $device --set-client-id wds,$cid_4 --get-current-settings)"
455 json_get_var gateway_4 gateway
456 json_get_var dns1_4 dns1
457 json_get_var dns2_4 dns2
458 json_get_var subnet_4 subnet
460 proto_init_update
"$ifname" 1
462 proto_add_ipv4_address
"$ip_4" "$subnet_4"
463 proto_add_ipv4_route
"$gateway_4" "128"
464 [ "$defaultroute" = 0 ] || proto_add_ipv4_route
"0.0.0.0" 0 "$gateway_4"
465 [ "$peerdns" = 0 ] ||
{
466 proto_add_dns_server
"$dns1_4"
467 proto_add_dns_server
"$dns2_4"
471 json_add_string zone
"$zone"
474 proto_send_update
"$interface"
477 json_add_string name
"${interface}_4"
478 json_add_string ifname
"@$interface"
479 json_add_string proto
"dhcp"
480 [ -n "$ip4table" ] && json_add_string ip4table
"$ip4table"
481 proto_add_dynamic_defaults
482 [ -n "$zone" ] && json_add_string zone
"$zone"
484 ubus call network add_dynamic
"$(json_dump)"
493 [ -n "$cid" ] ||
return
495 uqmi
-s -d "$device" --set-client-id wds
,"$cid" \
496 --stop-network 0xffffffff \
497 --autoconnect > /dev
/null
2>&1
500 uqmi
-s -d "$device" --set-client-id wds
,"$cid" \
501 --stop-network "$pdh" > /dev
/null
2>&1
504 uqmi
-s -d "$device" --set-client-id wds
,"$cid" \
505 --release-client-id wds
> /dev
/null
2>&1
508 proto_qmi_teardown
() {
511 local device cid_4 pdh_4 cid_6 pdh_6
514 [ -n "$ctl_device" ] && device
=$ctl_device
516 echo "Stopping network $interface"
518 json_load
"$(ubus call network.interface.$interface status)"
520 json_get_vars cid_4 pdh_4 cid_6 pdh_6
522 qmi_wds_stop
"$cid_4" "$pdh_4"
523 qmi_wds_stop
"$cid_6" "$pdh_6"
525 proto_init_update
"*" 0
526 proto_send_update
"$interface"
529 [ -n "$INCLUDE_ONLY" ] ||
{