2 * uqmi -- tiny QMI support implementation
4 * Copyright (C) 2014-2015 Felix Fietkau <nbd@openwrt.org>
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the
18 * Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
19 * Boston, MA 02110-1301 USA.
22 #include "qmi-message.h"
24 static struct qmi_nas_get_tx_rx_info_request tx_rx_req
;
25 static struct qmi_nas_set_system_selection_preference_request sel_req
;
32 print_earfcn_info(uint32_t earfcn
)
34 /* https://www.sqimway.com/lte_band.php */
42 { 0, 599, 1, 2100, "FDD" },
43 { 600, 1199, 2, 1800, "FDD" },
44 { 1200, 1949, 3, 1800, "FDD" },
45 { 1950, 2399, 4, 1700, "FDD" },
46 { 2400, 2649, 5, 850, "FDD" },
47 { 2650, 2749, 6, 800, "FDD" },
48 { 2750, 3449, 7, 2600, "FDD" },
49 { 3450, 3799, 8, 900, "FDD" },
50 { 3800, 4149, 9, 1800, "FDD" },
51 { 4150, 4749, 10, 1700, "FDD" },
52 { 4750, 4999, 11, 1500, "FDD" },
53 { 5000, 5179, 12, 700, "FDD" },
54 { 5180, 5279, 13, 700, "FDD" },
55 { 5280, 5379, 14, 700, "FDD" },
56 { 5730, 5849, 17, 700, "FDD" },
57 { 5850, 5999, 18, 850, "FDD" },
58 { 6000, 6149, 19, 850, "FDD" },
59 { 6150, 6449, 20, 800, "FDD" },
60 { 6450, 6599, 21, 1500, "FDD" },
61 { 6600, 7399, 22, 3500, "FDD" },
62 { 7500, 7699, 23, 2000, "FDD" },
63 { 7700, 8039, 24, 1600, "FDD" },
64 { 8040, 8689, 25, 1900, "FDD" },
65 { 8690, 9039, 26, 850, "FDD" },
66 { 9040, 9209, 27, 800, "FDD" },
67 { 9210, 9659, 28, 700, "FDD" },
68 { 9660, 9769, 29, 700, "SDL" },
69 { 9770, 9869, 30, 2300, "FDD" },
70 { 9870, 9919, 31, 450, "FDD" },
71 { 9920, 10359, 32, 1500, "SDL" },
72 { 36000, 36199, 33, 1900, "TDD" },
73 { 36200, 36349, 34, 2000, "TDD" },
74 { 36350, 36949, 35, 1900, "TDD" },
75 { 36950, 37549, 36, 1900, "TDD" },
76 { 37550, 37749, 37, 1900, "TDD" },
77 { 37750, 38249, 38, 2600, "TDD" },
78 { 38250, 38649, 39, 1900, "TDD" },
79 { 38650, 39649, 40, 2300, "TDD" },
80 { 39650, 41589, 41, 2500, "TDD" },
81 { 41590, 43589, 42, 3500, "TDD" },
82 { 43590, 45589, 43, 3700, "TDD" },
83 { 45590, 46589, 44, 700, "TDD" },
86 for (int i
= 0; i
< (sizeof(earfcn_ranges
) / sizeof(*earfcn_ranges
)); i
++) {
87 if (earfcn
<= earfcn_ranges
[i
].max
&& earfcn
>= earfcn_ranges
[i
].min
) {
88 blobmsg_add_u32(&status
, "band", earfcn_ranges
[i
].band
);
89 blobmsg_add_u32(&status
, "frequency", earfcn_ranges
[i
].freq
);
90 blobmsg_add_string(&status
, "duplex", earfcn_ranges
[i
].duplex
);
96 #define cmd_nas_do_set_system_selection_cb no_cb
97 static enum qmi_cmd_result
98 cmd_nas_do_set_system_selection_prepare(struct qmi_dev
*qmi
, struct qmi_request
*req
, struct qmi_msg
*msg
, char *arg
)
100 qmi_set_nas_set_system_selection_preference_request(msg
, &sel_req
);
101 return QMI_CMD_REQUEST
;
104 static enum qmi_cmd_result
107 static bool use_sel_req
= false;
111 uqmi_add_command(NULL
, __UQMI_COMMAND_nas_do_set_system_selection
);
117 #define cmd_nas_set_network_modes_cb no_cb
118 static enum qmi_cmd_result
119 cmd_nas_set_network_modes_prepare(struct qmi_dev
*qmi
, struct qmi_request
*req
, struct qmi_msg
*msg
, char *arg
)
121 static const struct {
123 QmiNasRatModePreference val
;
125 { "cdma", QMI_NAS_RAT_MODE_PREFERENCE_CDMA_1X
| QMI_NAS_RAT_MODE_PREFERENCE_CDMA_1XEVDO
},
126 { "td-scdma", QMI_NAS_RAT_MODE_PREFERENCE_TD_SCDMA
},
127 { "gsm", QMI_NAS_RAT_MODE_PREFERENCE_GSM
},
128 { "umts", QMI_NAS_RAT_MODE_PREFERENCE_UMTS
},
129 { "lte", QMI_NAS_RAT_MODE_PREFERENCE_LTE
},
131 QmiNasRatModePreference val
= 0;
135 for (word
= strtok(arg
, ",");
137 word
= strtok(NULL
, ",")) {
140 for (i
= 0; i
< ARRAY_SIZE(modes
); i
++) {
141 if (strcmp(word
, modes
[i
].name
) != 0 &&
142 strcmp(word
, "all") != 0)
150 uqmi_add_error("Invalid network mode");
155 qmi_set(&sel_req
, mode_preference
, val
);
156 return do_sel_network();
159 #define cmd_nas_set_network_preference_cb no_cb
160 static enum qmi_cmd_result
161 cmd_nas_set_network_preference_prepare(struct qmi_dev
*qmi
, struct qmi_request
*req
, struct qmi_msg
*msg
, char *arg
)
163 QmiNasGsmWcdmaAcquisitionOrderPreference pref
= QMI_NAS_GSM_WCDMA_ACQUISITION_ORDER_PREFERENCE_AUTOMATIC
;
165 if (!strcmp(arg
, "gsm"))
166 pref
= QMI_NAS_GSM_WCDMA_ACQUISITION_ORDER_PREFERENCE_GSM
;
167 else if (!strcmp(arg
, "wcdma"))
168 pref
= QMI_NAS_GSM_WCDMA_ACQUISITION_ORDER_PREFERENCE_WCDMA
;
170 qmi_set(&sel_req
, gsm_wcdma_acquisition_order_preference
, pref
);
171 return do_sel_network();
174 #define cmd_nas_set_roaming_cb no_cb
175 static enum qmi_cmd_result
176 cmd_nas_set_roaming_prepare(struct qmi_dev
*qmi
, struct qmi_request
*req
, struct qmi_msg
*msg
, char *arg
)
178 QmiNasRoamingPreference pref
;
180 if (!strcmp(arg
, "any"))
181 pref
= QMI_NAS_ROAMING_PREFERENCE_ANY
;
182 else if (!strcmp(arg
, "only"))
183 pref
= QMI_NAS_ROAMING_PREFERENCE_NOT_OFF
;
184 else if (!strcmp(arg
, "off"))
185 pref
= QMI_NAS_ROAMING_PREFERENCE_OFF
;
187 return uqmi_add_error("Invalid argument");
189 qmi_set(&sel_req
, roaming_preference
, pref
);
190 return do_sel_network();
193 #define cmd_nas_set_mcc_cb no_cb
194 static enum qmi_cmd_result
195 cmd_nas_set_mcc_prepare(struct qmi_dev
*qmi
, struct qmi_request
*req
, struct qmi_msg
*msg
, char *arg
)
198 int value
= strtoul(arg
, &err
, 10);
200 uqmi_add_error("Invalid MCC value");
204 sel_req
.data
.network_selection_preference
.mcc
= value
;
205 plmn_code_flag
.mcc_is_set
= true;
209 #define cmd_nas_set_mnc_cb no_cb
210 static enum qmi_cmd_result
211 cmd_nas_set_mnc_prepare(struct qmi_dev
*qmi
, struct qmi_request
*req
, struct qmi_msg
*msg
, char *arg
)
214 int value
= strtoul(arg
, &err
, 10);
216 uqmi_add_error("Invalid MNC value");
220 sel_req
.data
.network_selection_preference
.mnc
= value
;
221 plmn_code_flag
.mnc_is_set
= true;
225 #define cmd_nas_set_plmn_cb no_cb
226 static enum qmi_cmd_result
227 cmd_nas_set_plmn_prepare(struct qmi_dev
*qmi
, struct qmi_request
*req
, struct qmi_msg
*msg
, char *arg
)
229 sel_req
.set
.network_selection_preference
= 1;
230 sel_req
.data
.network_selection_preference
.mode
= QMI_NAS_NETWORK_SELECTION_PREFERENCE_AUTOMATIC
;
232 if (!plmn_code_flag
.mcc_is_set
&& plmn_code_flag
.mnc_is_set
) {
233 uqmi_add_error("No MCC value");
237 if (plmn_code_flag
.mcc_is_set
&& sel_req
.data
.network_selection_preference
.mcc
) {
238 if (!plmn_code_flag
.mnc_is_set
) {
239 uqmi_add_error("No MNC value");
242 sel_req
.data
.network_selection_preference
.mode
= QMI_NAS_NETWORK_SELECTION_PREFERENCE_MANUAL
;
246 return do_sel_network();
249 #define cmd_nas_initiate_network_register_cb no_cb
250 static enum qmi_cmd_result
251 cmd_nas_initiate_network_register_prepare(struct qmi_dev
*qmi
, struct qmi_request
*req
, struct qmi_msg
*msg
, char *arg
)
253 static struct qmi_nas_initiate_network_register_request register_req
= {
254 QMI_INIT(action
, QMI_NAS_NETWORK_REGISTER_TYPE_AUTOMATIC
)
257 qmi_set_nas_initiate_network_register_request(msg
, ®ister_req
);
258 return QMI_CMD_REQUEST
;
262 cmd_nas_get_signal_info_cb(struct qmi_dev
*qmi
, struct qmi_request
*req
, struct qmi_msg
*msg
)
264 struct qmi_nas_get_signal_info_response res
;
267 qmi_parse_nas_get_signal_info_response(msg
, &res
);
269 c
= blobmsg_open_table(&status
, NULL
);
270 if (res
.set
.cdma_signal_strength
) {
271 blobmsg_add_string(&status
, "type", "cdma");
272 blobmsg_add_u32(&status
, "rssi", (int32_t) res
.data
.cdma_signal_strength
.rssi
);
273 blobmsg_add_u32(&status
, "ecio", (int32_t) res
.data
.cdma_signal_strength
.ecio
);
276 if (res
.set
.hdr_signal_strength
) {
277 blobmsg_add_string(&status
, "type", "hdr");
278 blobmsg_add_u32(&status
, "rssi", (int32_t) res
.data
.hdr_signal_strength
.rssi
);
279 blobmsg_add_u32(&status
, "ecio", (int32_t) res
.data
.hdr_signal_strength
.ecio
);
280 blobmsg_add_u32(&status
, "io", res
.data
.hdr_signal_strength
.io
);
283 if (res
.set
.gsm_signal_strength
) {
284 blobmsg_add_string(&status
, "type", "gsm");
285 blobmsg_add_u32(&status
, "signal", (int32_t) res
.data
.gsm_signal_strength
);
288 if (res
.set
.wcdma_signal_strength
) {
289 blobmsg_add_string(&status
, "type", "wcdma");
290 blobmsg_add_u32(&status
, "rssi", (int32_t) res
.data
.wcdma_signal_strength
.rssi
);
291 blobmsg_add_u32(&status
, "ecio", (int32_t) res
.data
.wcdma_signal_strength
.ecio
);
294 if (res
.set
.lte_signal_strength
) {
295 blobmsg_add_string(&status
, "type", "lte");
296 blobmsg_add_u32(&status
, "rssi", (int32_t) res
.data
.lte_signal_strength
.rssi
);
297 blobmsg_add_u32(&status
, "rsrq", (int32_t) res
.data
.lte_signal_strength
.rsrq
);
298 blobmsg_add_u32(&status
, "rsrp", (int32_t) res
.data
.lte_signal_strength
.rsrp
);
299 blobmsg_add_double(&status
, "snr", (double) res
.data
.lte_signal_strength
.snr
*0.1);
302 if (res
.set
.tdma_signal_strength
) {
303 blobmsg_add_string(&status
, "type", "tdma");
304 blobmsg_add_u32(&status
, "signal", (int32_t) res
.data
.tdma_signal_strength
);
307 blobmsg_close_table(&status
, c
);
311 print_system_info(uint8_t svc_status
, uint8_t tsvc_status
, bool preferred
, bool system_info
,
312 bool domain_valid
, uint8_t domain
,
313 bool service_cap_valid
, uint8_t service_cap
,
314 bool roaming_status_valid
, uint8_t roaming_status
,
315 bool forbidden_valid
, bool forbidden
,
316 bool network_id_valid
, char *mcc
, char *mnc
,
317 bool lac_valid
, uint16_t lac
)
319 static const char *map_service
[] = {
320 [QMI_NAS_SERVICE_STATUS_NONE
] = "none",
321 [QMI_NAS_SERVICE_STATUS_LIMITED
] = "limited",
322 [QMI_NAS_SERVICE_STATUS_AVAILABLE
] = "available",
323 [QMI_NAS_SERVICE_STATUS_LIMITED_REGIONAL
] = "limited regional",
324 [QMI_NAS_SERVICE_STATUS_POWER_SAVE
] = "power save",
327 static const char *map_roaming
[] = {
328 [QMI_NAS_ROAMING_STATUS_OFF
] = "off",
329 [QMI_NAS_ROAMING_STATUS_ON
] = "on",
330 [QMI_NAS_ROAMING_STATUS_BLINK
] = "blink",
331 [QMI_NAS_ROAMING_STATUS_OUT_OF_NEIGHBORHOOD
] = "out of neighborhood",
332 [QMI_NAS_ROAMING_STATUS_OUT_OF_BUILDING
] = "out of building",
333 [QMI_NAS_ROAMING_STATUS_PREFERRED_SYSTEM
] = "preferred system",
334 [QMI_NAS_ROAMING_STATUS_AVAILABLE_SYSTEM
] = "available system",
335 [QMI_NAS_ROAMING_STATUS_ALLIANCE_PARTNER
] = "alliance partner",
336 [QMI_NAS_ROAMING_STATUS_PREMIUM_PARTNER
] = "premium partner",
337 [QMI_NAS_ROAMING_STATUS_FULL_SERVICE
] = "full service",
338 [QMI_NAS_ROAMING_STATUS_PARTIAL_SERVICE
] = "partial service",
339 [QMI_NAS_ROAMING_STATUS_BANNER_ON
] = "banner on",
340 [QMI_NAS_ROAMING_STATUS_BANNER_OFF
] = "banner off",
343 static const char *map_network
[] = {
344 [QMI_NAS_NETWORK_SERVICE_DOMAIN_NONE
] = "none",
345 [QMI_NAS_NETWORK_SERVICE_DOMAIN_CS
] = "cs",
346 [QMI_NAS_NETWORK_SERVICE_DOMAIN_PS
] = "ps",
347 [QMI_NAS_NETWORK_SERVICE_DOMAIN_CS_PS
] = "cs-ps",
348 [QMI_NAS_NETWORK_SERVICE_DOMAIN_UNKNOWN
] = "unknown",
351 blobmsg_add_string(&status
, "service_status", map_service
[svc_status
]);
352 blobmsg_add_string(&status
, "true_service_status", map_service
[tsvc_status
]);
353 blobmsg_add_u8(&status
, "preferred_data_path", preferred
);
357 blobmsg_add_string(&status
, "domain", map_network
[domain
]);
358 if (service_cap_valid
)
359 blobmsg_add_string(&status
, "service", map_network
[service_cap
]);
360 if (roaming_status_valid
)
361 blobmsg_add_string(&status
, "roaming_status", map_roaming
[roaming_status
]);
363 blobmsg_add_u8(&status
, "forbidden", forbidden
);
364 if (network_id_valid
) {
365 blobmsg_add_string(&status
, "mcc", mcc
);
366 if ((uint8_t)mnc
[2] == 255)
368 blobmsg_add_string(&status
, "mnc", mnc
);
371 blobmsg_add_u32(&status
, "location_area_code", (int32_t) lac
);
376 cmd_nas_get_system_info_cb(struct qmi_dev
*qmi
, struct qmi_request
*req
, struct qmi_msg
*msg
)
378 static const char *cell_status
[] = {
379 [QMI_NAS_CELL_ACCESS_STATUS_NORMAL_ONLY
] = "normal",
380 [QMI_NAS_CELL_ACCESS_STATUS_EMERGENCY_ONLY
] = "emergency",
381 [QMI_NAS_CELL_ACCESS_STATUS_NO_CALLS
] = "no calls",
382 [QMI_NAS_CELL_ACCESS_STATUS_ALL_CALLS
] = "all calls",
383 [QMI_NAS_CELL_ACCESS_STATUS_UNKNOWN
] = "unknown",
386 struct qmi_nas_get_system_info_response res
;
389 qmi_parse_nas_get_system_info_response(msg
, &res
);
390 t
= blobmsg_open_table(&status
, NULL
);
391 if (res
.set
.gsm_service_status
) {
392 c
= blobmsg_open_table(&status
, "gsm");
393 print_system_info(res
.data
.gsm_service_status
.service_status
,
394 res
.data
.gsm_service_status
.true_service_status
,
395 res
.data
.gsm_service_status
.preferred_data_path
,
396 res
.set
.gsm_system_info_v2
,
397 res
.data
.gsm_system_info_v2
.domain_valid
,
398 res
.data
.gsm_system_info_v2
.domain
,
399 res
.data
.gsm_system_info_v2
.service_capability_valid
,
400 res
.data
.gsm_system_info_v2
.service_capability
,
401 res
.data
.gsm_system_info_v2
.roaming_status_valid
,
402 res
.data
.gsm_system_info_v2
.roaming_status
,
403 res
.data
.gsm_system_info_v2
.forbidden_valid
,
404 res
.data
.gsm_system_info_v2
.forbidden
,
405 res
.data
.gsm_system_info_v2
.network_id_valid
,
406 res
.data
.gsm_system_info_v2
.mcc
,
407 res
.data
.gsm_system_info_v2
.mnc
,
408 res
.data
.gsm_system_info_v2
.lac_valid
,
409 res
.data
.gsm_system_info_v2
.lac
);
410 if (res
.set
.gsm_system_info_v2
&& res
.data
.gsm_system_info_v2
.cid_valid
)
411 blobmsg_add_u32(&status
, "cell_id",
412 res
.data
.gsm_system_info_v2
.cid
);
413 if (res
.set
.additional_gsm_system_info
&&
414 res
.data
.additional_gsm_system_info
.geo_system_index
!= 0xFFFF)
415 blobmsg_add_u32(&status
, "geo_system_index",
416 res
.data
.additional_gsm_system_info
.geo_system_index
);
417 blobmsg_close_table(&status
, c
);
420 if (res
.set
.wcdma_service_status
) {
421 c
= blobmsg_open_table(&status
, "wcdma");
422 print_system_info(res
.data
.wcdma_service_status
.service_status
,
423 res
.data
.wcdma_service_status
.true_service_status
,
424 res
.data
.wcdma_service_status
.preferred_data_path
,
425 res
.set
.wcdma_system_info_v2
,
426 res
.data
.wcdma_system_info_v2
.domain_valid
,
427 res
.data
.wcdma_system_info_v2
.domain
,
428 res
.data
.wcdma_system_info_v2
.service_capability_valid
,
429 res
.data
.wcdma_system_info_v2
.service_capability
,
430 res
.data
.wcdma_system_info_v2
.roaming_status_valid
,
431 res
.data
.wcdma_system_info_v2
.roaming_status
,
432 res
.data
.wcdma_system_info_v2
.forbidden_valid
,
433 res
.data
.wcdma_system_info_v2
.forbidden
,
434 res
.data
.wcdma_system_info_v2
.network_id_valid
,
435 res
.data
.wcdma_system_info_v2
.mcc
,
436 res
.data
.wcdma_system_info_v2
.mnc
,
437 res
.data
.wcdma_system_info_v2
.lac_valid
,
438 res
.data
.wcdma_system_info_v2
.lac
);
439 if (res
.set
.wcdma_system_info_v2
&& res
.data
.wcdma_system_info_v2
.cid_valid
) {
440 blobmsg_add_u32(&status
, "rnc_id",res
.data
.wcdma_system_info_v2
.cid
/65536);
441 blobmsg_add_u32(&status
, "cell_id",res
.data
.wcdma_system_info_v2
.cid
%65536);
443 if (res
.set
.additional_wcdma_system_info
&&
444 res
.data
.additional_wcdma_system_info
.geo_system_index
!= 0xFFFF)
445 blobmsg_add_u32(&status
, "geo_system_index",
446 res
.data
.additional_wcdma_system_info
.geo_system_index
);
447 blobmsg_close_table(&status
, c
);
450 if (res
.set
.lte_service_status
) {
451 c
= blobmsg_open_table(&status
, "lte");
452 print_system_info(res
.data
.lte_service_status
.service_status
,
453 res
.data
.lte_service_status
.true_service_status
,
454 res
.data
.lte_service_status
.preferred_data_path
,
455 res
.set
.lte_system_info_v2
,
456 res
.data
.lte_system_info_v2
.domain_valid
,
457 res
.data
.lte_system_info_v2
.domain
,
458 res
.data
.lte_system_info_v2
.service_capability_valid
,
459 res
.data
.lte_system_info_v2
.service_capability
,
460 res
.data
.lte_system_info_v2
.roaming_status_valid
,
461 res
.data
.lte_system_info_v2
.roaming_status
,
462 res
.data
.lte_system_info_v2
.forbidden_valid
,
463 res
.data
.lte_system_info_v2
.forbidden
,
464 res
.data
.lte_system_info_v2
.network_id_valid
,
465 res
.data
.lte_system_info_v2
.mcc
,
466 res
.data
.lte_system_info_v2
.mnc
,
467 res
.data
.lte_system_info_v2
.lac_valid
,
468 res
.data
.lte_system_info_v2
.lac
);
469 if (res
.set
.lte_system_info_v2
&& res
.data
.lte_system_info_v2
.tac_valid
)
470 blobmsg_add_u32(&status
, "tracking_area_code",
471 res
.data
.lte_system_info_v2
.tac
);
472 if (res
.set
.lte_system_info_v2
&& res
.data
.lte_system_info_v2
.cid_valid
) {
473 blobmsg_add_u32(&status
, "enodeb_id",res
.data
.lte_system_info_v2
.cid
/256);
474 blobmsg_add_u32(&status
, "cell_id",res
.data
.lte_system_info_v2
.cid
%256);
476 if (res
.set
.additional_lte_system_info
&&
477 res
.data
.additional_lte_system_info
.geo_system_index
!= 0xFFFF)
478 blobmsg_add_u32(&status
, "geo_system_index",
479 res
.data
.additional_lte_system_info
.geo_system_index
);
480 if (res
.set
.lte_voice_support
)
481 blobmsg_add_u8(&status
, "voice_support", res
.data
.lte_voice_support
);
482 if (res
.set
.ims_voice_support
)
483 blobmsg_add_u8(&status
, "ims_voice_support", res
.data
.ims_voice_support
);
484 if (res
.set
.lte_cell_access_status
)
485 blobmsg_add_string(&status
, "cell_access_status",
486 cell_status
[res
.data
.lte_cell_access_status
]);
487 if (res
.set
.network_selection_registration_restriction
)
488 blobmsg_add_u32(&status
, "registration_restriction",
489 res
.data
.network_selection_registration_restriction
);
490 if (res
.set
.lte_registration_domain
)
491 blobmsg_add_u32(&status
, "registration_domain",
492 res
.data
.lte_registration_domain
);
493 if (res
.set
.eutra_with_nr5g_availability
)
494 blobmsg_add_u8(&status
, "5g_nsa_available",
495 res
.data
.eutra_with_nr5g_availability
);
496 if (res
.set
.dcnr_restriction_info
)
497 blobmsg_add_u8(&status
, "dcnr_restriction", res
.data
.dcnr_restriction_info
);
499 blobmsg_close_table(&status
, c
);
501 blobmsg_close_table(&status
, t
);
504 static enum qmi_cmd_result
505 cmd_nas_get_system_info_prepare(struct qmi_dev
*qmi
, struct qmi_request
*req
, struct qmi_msg
*msg
, char *arg
)
507 qmi_set_nas_get_system_info_request(msg
);
508 return QMI_CMD_REQUEST
;
512 print_channel_info(int32_t cell_id
, int32_t channel
, uint32_t bw
)
514 static const char *map_bandwidth
[] = {
515 [QMI_NAS_DL_BANDWIDTH_1_4
] = "1.4",
516 [QMI_NAS_DL_BANDWIDTH_3
] = "3",
517 [QMI_NAS_DL_BANDWIDTH_5
] = "5",
518 [QMI_NAS_DL_BANDWIDTH_10
] = "10",
519 [QMI_NAS_DL_BANDWIDTH_15
] = "15",
520 [QMI_NAS_DL_BANDWIDTH_20
] = "20",
521 [QMI_NAS_DL_BANDWIDTH_INVALID
] = "invalid",
522 [QMI_NAS_DL_BANDWIDTH_UNKNOWN
] = "unknown",
525 blobmsg_add_u32(&status
, "cell_id", cell_id
);
526 blobmsg_add_u32(&status
, "channel", channel
);
527 print_earfcn_info(channel
);
528 blobmsg_add_string(&status
, "bandwidth", map_bandwidth
[bw
]);
532 cmd_nas_get_lte_cphy_ca_info_cb(struct qmi_dev
*qmi
, struct qmi_request
*req
, struct qmi_msg
*msg
)
534 struct qmi_nas_get_lte_cphy_ca_info_response res
;
535 static const char *scell_state
[] = {
536 [QMI_NAS_SCELL_STATE_DECONFIGURED
] = "deconfigured",
537 [QMI_NAS_SCELL_STATE_DEACTIVATED
] = "deactivated",
538 [QMI_NAS_SCELL_STATE_ACTIVATED
] = "activated",
544 qmi_parse_nas_get_lte_cphy_ca_info_response(msg
, &res
);
545 t
= blobmsg_open_table(&status
, NULL
);
546 if (res
.set
.phy_ca_agg_pcell_info
) {
547 c
= blobmsg_open_table(&status
, "primary");
548 print_channel_info(res
.data
.phy_ca_agg_pcell_info
.physical_cell_id
,
549 res
.data
.phy_ca_agg_pcell_info
.rx_channel
,
550 res
.data
.phy_ca_agg_pcell_info
.dl_bandwidth
);
551 blobmsg_close_table(&status
, c
);
553 if (res
.set
.phy_ca_agg_scell_info
&& res
.data
.phy_ca_agg_secondary_cells_n
) {
554 for (i
= 0; i
< res
.data
.phy_ca_agg_secondary_cells_n
; i
++) {
555 if (res
.data
.phy_ca_agg_secondary_cells
[i
].rx_channel
== 0)
557 sprintf(idx_buf
, "secondary_%d",
558 res
.data
.phy_ca_agg_secondary_cells
[i
].cell_index
);
559 c
= blobmsg_open_table(&status
, idx_buf
);
560 print_channel_info(res
.data
.phy_ca_agg_secondary_cells
[i
].physical_cell_id
,
561 res
.data
.phy_ca_agg_secondary_cells
[i
].rx_channel
,
562 res
.data
.phy_ca_agg_secondary_cells
[i
].dl_bandwidth
);
563 blobmsg_add_string(&status
, "state",
564 scell_state
[res
.data
.phy_ca_agg_secondary_cells
[i
].state
]);
565 blobmsg_close_table(&status
, c
);
568 if (res
.set
.scell_index
)
569 sprintf(idx_buf
, "secondary_%d", res
.data
.scell_index
);
571 sprintf(idx_buf
, "secondary");
572 if (res
.set
.phy_ca_agg_scell_info
&& res
.data
.phy_ca_agg_scell_info
.rx_channel
!= 0) {
573 c
= blobmsg_open_table(&status
, idx_buf
);
574 print_channel_info(res
.data
.phy_ca_agg_scell_info
.physical_cell_id
,
575 res
.data
.phy_ca_agg_scell_info
.rx_channel
,
576 res
.data
.phy_ca_agg_scell_info
.dl_bandwidth
);
577 blobmsg_add_string(&status
, "state",
578 scell_state
[res
.data
.phy_ca_agg_scell_info
.state
]);
579 blobmsg_close_table(&status
, c
);
582 blobmsg_close_table(&status
, t
);
585 static enum qmi_cmd_result
586 cmd_nas_get_lte_cphy_ca_info_prepare(struct qmi_dev
*qmi
, struct qmi_request
*req
, struct qmi_msg
*msg
, char *arg
)
588 qmi_set_nas_get_lte_cphy_ca_info_request(msg
);
589 return QMI_CMD_REQUEST
;
593 print_chain_info(int8_t radio
, bool tuned
, int32_t rssi
, int32_t ecio
, int32_t rsrp
, int32_t rscp
, uint32_t phase
)
595 blobmsg_add_u8(&status
, "tuned", tuned
);
596 blobmsg_add_double(&status
, "rssi", (double) rssi
*0.1);
597 if (radio
== QMI_NAS_RADIO_INTERFACE_LTE
) {
598 blobmsg_add_double(&status
, "rsrq", (double) ecio
*-0.1);
599 blobmsg_add_double(&status
, "rsrp", (double) rsrp
*-0.1);
601 if (radio
== QMI_NAS_RADIO_INTERFACE_UMTS
) {
602 blobmsg_add_double(&status
, "ecio", (double) ecio
*-0.1);
603 blobmsg_add_double(&status
, "rscp", (double) rscp
*-0.1);
605 if (phase
!= 0xFFFFFFFF)
606 blobmsg_add_double(&status
, "phase", (double) phase
*0.01);
610 cmd_nas_get_tx_rx_info_cb(struct qmi_dev
*qmi
, struct qmi_request
*req
, struct qmi_msg
*msg
)
612 struct qmi_nas_get_tx_rx_info_response res
;
615 qmi_parse_nas_get_tx_rx_info_response(msg
, &res
);
616 t
= blobmsg_open_table(&status
, NULL
);
617 if (res
.set
.rx_chain_0_info
) {
618 c
= blobmsg_open_table(&status
, "rx_chain_0");
619 print_chain_info(tx_rx_req
.data
.radio_interface
,
620 res
.data
.rx_chain_0_info
.is_radio_tuned
,
621 res
.data
.rx_chain_0_info
.rx_power
,
622 res
.data
.rx_chain_0_info
.ecio
,
623 res
.data
.rx_chain_0_info
.rsrp
,
624 res
.data
.rx_chain_0_info
.rscp
,
625 res
.data
.rx_chain_0_info
.phase
);
626 blobmsg_close_table(&status
, c
);
628 if (res
.set
.rx_chain_1_info
) {
629 c
= blobmsg_open_table(&status
, "rx_chain_1");
630 print_chain_info(tx_rx_req
.data
.radio_interface
,
631 res
.data
.rx_chain_1_info
.is_radio_tuned
,
632 res
.data
.rx_chain_1_info
.rx_power
,
633 res
.data
.rx_chain_1_info
.ecio
,
634 res
.data
.rx_chain_1_info
.rsrp
,
635 res
.data
.rx_chain_1_info
.rscp
,
636 res
.data
.rx_chain_1_info
.phase
);
637 blobmsg_close_table(&status
, c
);
639 if (res
.set
.rx_chain_2_info
) {
640 c
= blobmsg_open_table(&status
, "rx_chain_2");
641 print_chain_info(tx_rx_req
.data
.radio_interface
,
642 res
.data
.rx_chain_2_info
.is_radio_tuned
,
643 res
.data
.rx_chain_2_info
.rx_power
,
644 res
.data
.rx_chain_2_info
.ecio
,
645 res
.data
.rx_chain_2_info
.rsrp
,
646 res
.data
.rx_chain_2_info
.rscp
,
647 res
.data
.rx_chain_2_info
.phase
);
648 blobmsg_close_table(&status
, c
);
650 if (res
.set
.rx_chain_3_info
) {
651 c
= blobmsg_open_table(&status
, "rx_chain_3");
652 print_chain_info(tx_rx_req
.data
.radio_interface
,
653 res
.data
.rx_chain_3_info
.is_radio_tuned
,
654 res
.data
.rx_chain_3_info
.rx_power
,
655 res
.data
.rx_chain_3_info
.ecio
,
656 res
.data
.rx_chain_3_info
.rsrp
,
657 res
.data
.rx_chain_3_info
.rscp
,
658 res
.data
.rx_chain_3_info
.phase
);
659 blobmsg_close_table(&status
, c
);
661 if (res
.set
.tx_info
) {
662 c
= blobmsg_open_table(&status
, "tx");
663 blobmsg_add_u8(&status
, "traffic", res
.data
.tx_info
.is_in_traffic
);
664 if (res
.data
.tx_info
.is_in_traffic
)
665 blobmsg_add_double(&status
, "tx_power",
666 (double) res
.data
.tx_info
.tx_power
*0.1);
667 blobmsg_close_table(&status
, c
);
669 blobmsg_close_table(&status
, t
);
673 static enum qmi_cmd_result
674 cmd_nas_get_tx_rx_info_prepare(struct qmi_dev
*qmi
, struct qmi_request
*req
, struct qmi_msg
*msg
, char *arg
)
678 if (!strcmp(arg
, "lte"))
679 radio
= QMI_NAS_RADIO_INTERFACE_LTE
;
680 else if (!strcmp(arg
, "umts"))
681 radio
= QMI_NAS_RADIO_INTERFACE_UMTS
;
682 else if (!strcmp(arg
, "gsm"))
683 radio
= QMI_NAS_RADIO_INTERFACE_GSM
;
685 return uqmi_add_error("Invalid argument");
687 qmi_set(&tx_rx_req
, radio_interface
, radio
);
688 qmi_set_nas_get_tx_rx_info_request(msg
, &tx_rx_req
);
689 return QMI_CMD_REQUEST
;
693 print_lte_info(int32_t cell_id
, int16_t rsrp
, int16_t rsrq
, int16_t rssi
)
695 blobmsg_add_u32(&status
, "physical_cell_id", cell_id
);
696 blobmsg_add_double(&status
, "rsrq", ((double)rsrq
)/10);
697 blobmsg_add_double(&status
, "rsrp", ((double)rsrp
)/10);
698 blobmsg_add_double(&status
, "rssi", ((double)rssi
)/10);
702 print_sel_info(int32_t priority
, int32_t high
, int32_t low
)
704 blobmsg_add_u32(&status
, "cell_reselection_priority", priority
);
705 blobmsg_add_u32(&status
, "cell_reselection_low", low
);
706 blobmsg_add_u32(&status
, "cell_reselection_high", high
);
710 cmd_nas_get_cell_location_info_cb(struct qmi_dev
*qmi
, struct qmi_request
*req
, struct qmi_msg
*msg
)
712 struct qmi_nas_get_cell_location_info_response res
;
713 void *c
, *t
, *cell
, *freq
;
716 qmi_parse_nas_get_cell_location_info_response(msg
, &res
);
717 t
= blobmsg_open_table(&status
, NULL
);
719 if (res
.set
.umts_info_v2
) {
720 c
= blobmsg_open_table(&status
, "umts_info");
721 blobmsg_add_u32(&status
, "location_area_code", res
.data
.umts_info_v2
.lac
);
722 blobmsg_add_u32(&status
, "cell_id", res
.data
.umts_info_v2
.cell_id
);
723 blobmsg_add_u32(&status
, "channel",
724 res
.data
.umts_info_v2
.utra_absolute_rf_channel_number
);
725 blobmsg_add_u32(&status
, "primary_scrambling_code",
726 res
.data
.umts_info_v2
.primary_scrambling_code
);
727 blobmsg_add_u32(&status
, "rscp", res
.data
.umts_info_v2
.rscp
);
728 blobmsg_add_u32(&status
, "ecio", res
.data
.umts_info_v2
.ecio
);
729 for (j
= 0; j
< res
.data
.umts_info_v2
.cell_n
; j
++) {
730 cell
= blobmsg_open_table(&status
, NULL
);
731 blobmsg_add_u32(&status
, "channel",
732 res
.data
.umts_info_v2
.cell
[j
].utra_absolute_rf_channel_number
);
733 blobmsg_add_u32(&status
, "primary_scrambling_code",
734 res
.data
.umts_info_v2
.cell
[j
].primary_scrambling_code
);
735 blobmsg_add_u32(&status
, "rscp", res
.data
.umts_info_v2
.cell
[j
].rscp
);
736 blobmsg_add_u32(&status
, "ecio", res
.data
.umts_info_v2
.cell
[j
].ecio
);
737 blobmsg_close_table(&status
, cell
);
739 for (j
= 0; j
< res
.data
.umts_info_v2
.neighboring_geran_n
; j
++) {
740 cell
= blobmsg_open_table(&status
, "neighboring_geran");
741 blobmsg_add_u32(&status
, "channel",
742 res
.data
.umts_info_v2
.neighboring_geran
[j
].geran_absolute_rf_channel_number
);
743 blobmsg_add_u8(&status
, "network_color_code",
744 res
.data
.umts_info_v2
.neighboring_geran
[j
].network_color_code
);
745 blobmsg_add_u8(&status
, "base_station_color_code",
746 res
.data
.umts_info_v2
.neighboring_geran
[j
].base_station_color_code
);
747 blobmsg_add_u32(&status
, "rssi",
748 res
.data
.umts_info_v2
.neighboring_geran
[j
].rssi
);
749 blobmsg_close_table(&status
, cell
);
751 blobmsg_close_table(&status
, c
);
753 if (res
.set
.intrafrequency_lte_info_v2
) {
754 c
= blobmsg_open_table(&status
, "intrafrequency_lte_info");
755 blobmsg_add_u32(&status
, "tracking_area_code",
756 res
.data
.intrafrequency_lte_info_v2
.tracking_area_code
);
757 blobmsg_add_u32(&status
, "enodeb_id",
758 res
.data
.intrafrequency_lte_info_v2
.global_cell_id
/256);
759 blobmsg_add_u32(&status
, "cell_id",
760 res
.data
.intrafrequency_lte_info_v2
.global_cell_id
%256);
761 blobmsg_add_u32(&status
, "channel",
762 res
.data
.intrafrequency_lte_info_v2
.eutra_absolute_rf_channel_number
);
763 print_earfcn_info(res
.data
.intrafrequency_lte_info_v2
.eutra_absolute_rf_channel_number
);
764 blobmsg_add_u32(&status
, "serving_cell_id",
765 res
.data
.intrafrequency_lte_info_v2
.serving_cell_id
);
766 if (res
.data
.intrafrequency_lte_info_v2
.ue_in_idle
) {
767 blobmsg_add_u32(&status
, "cell_reselection_priority",
768 res
.data
.intrafrequency_lte_info_v2
.cell_reselection_priority
);
769 blobmsg_add_u32(&status
, "s_non_intra_search_threshold",
770 res
.data
.intrafrequency_lte_info_v2
.s_non_intra_search_threshold
);
771 blobmsg_add_u32(&status
, "serving_cell_low_threshold",
772 res
.data
.intrafrequency_lte_info_v2
.serving_cell_low_threshold
);
773 blobmsg_add_u32(&status
, "s_intra_search_threshold",
774 res
.data
.intrafrequency_lte_info_v2
.s_intra_search_threshold
);
776 for (i
= 0; i
< res
.data
.intrafrequency_lte_info_v2
.cell_n
; i
++) {
777 cell
= blobmsg_open_table(&status
, NULL
);
778 print_lte_info(res
.data
.intrafrequency_lte_info_v2
.cell
[i
].physical_cell_id
,
779 res
.data
.intrafrequency_lte_info_v2
.cell
[i
].rsrq
,
780 res
.data
.intrafrequency_lte_info_v2
.cell
[i
].rsrp
,
781 res
.data
.intrafrequency_lte_info_v2
.cell
[i
].rssi
);
782 if (res
.data
.intrafrequency_lte_info_v2
.ue_in_idle
)
783 blobmsg_add_u32(&status
, "cell_selection_rx_level",
784 res
.data
.intrafrequency_lte_info_v2
.cell
[i
].cell_selection_rx_level
);
785 blobmsg_close_table(&status
, cell
);
787 blobmsg_close_table(&status
, c
);
789 if (res
.set
.interfrequency_lte_info
) {
790 if (res
.data
.interfrequency_lte_info
.frequency_n
> 0)
791 c
= blobmsg_open_table(&status
, "interfrequency_lte_info");
792 for (i
= 0; i
< res
.data
.interfrequency_lte_info
.frequency_n
; i
++) {
793 freq
= blobmsg_open_table(&status
, NULL
);
794 blobmsg_add_u32(&status
, "channel",
795 res
.data
.interfrequency_lte_info
.frequency
[i
].eutra_absolute_rf_channel_number
);
796 print_earfcn_info(res
.data
.interfrequency_lte_info
.frequency
[i
].eutra_absolute_rf_channel_number
);
797 if (res
.data
.interfrequency_lte_info
.ue_in_idle
) {
798 print_sel_info(res
.data
.interfrequency_lte_info
.frequency
[i
].cell_reselection_priority
,
799 res
.data
.interfrequency_lte_info
.frequency
[i
].cell_selection_rx_level_high_threshold
,
800 res
.data
.interfrequency_lte_info
.frequency
[i
].cell_selection_rx_level_low_threshold
);
802 for (j
= 0; j
< res
.data
.interfrequency_lte_info
.frequency
[i
].cell_n
; j
++) {
803 cell
= blobmsg_open_table(&status
, NULL
);
804 print_lte_info(res
.data
.interfrequency_lte_info
.frequency
[i
].cell
[j
].physical_cell_id
,
805 res
.data
.interfrequency_lte_info
.frequency
[i
].cell
[j
].rsrq
,
806 res
.data
.interfrequency_lte_info
.frequency
[i
].cell
[j
].rsrp
,
807 res
.data
.interfrequency_lte_info
.frequency
[i
].cell
[j
].rssi
);
808 if (res
.data
.interfrequency_lte_info
.ue_in_idle
)
809 blobmsg_add_u32(&status
, "cell_selection_rx_level",
810 res
.data
.interfrequency_lte_info
.frequency
[i
].cell
[j
].cell_selection_rx_level
);
811 blobmsg_close_table(&status
, cell
);
813 blobmsg_close_table(&status
, freq
);
815 if (res
.data
.interfrequency_lte_info
.frequency_n
> 0)
816 blobmsg_close_table(&status
, c
);
818 if (res
.set
.lte_info_neighboring_gsm
) {
819 if (res
.data
.lte_info_neighboring_gsm
.frequency_n
> 0)
820 c
= blobmsg_open_table(&status
, "lte_info_neighboring_gsm");
821 for (i
= 0; i
< res
.data
.lte_info_neighboring_gsm
.frequency_n
; i
++) {
822 freq
= blobmsg_open_table(&status
, NULL
);
823 blobmsg_add_u32(&status
, "ncc_permitted",
824 res
.data
.lte_info_neighboring_gsm
.frequency
[i
].ncc_permitted
);
825 if (res
.data
.lte_info_neighboring_gsm
.ue_in_idle
) {
826 print_sel_info(res
.data
.lte_info_neighboring_gsm
.frequency
[i
].cell_reselection_priority
,
827 res
.data
.lte_info_neighboring_gsm
.frequency
[i
].cell_reselection_high_threshold
,
828 res
.data
.lte_info_neighboring_gsm
.frequency
[i
].cell_reselection_low_threshold
);
830 for (j
= 0; j
< res
.data
.lte_info_neighboring_gsm
.frequency
[i
].cell_n
; j
++) {
831 cell
= blobmsg_open_table(&status
, NULL
);
832 blobmsg_add_u32(&status
, "channel",
833 res
.data
.lte_info_neighboring_gsm
.frequency
[i
].cell
[j
].geran_absolute_rf_channel_number
);
834 blobmsg_add_u32(&status
, "base_station_identity_code",
835 res
.data
.lte_info_neighboring_gsm
.frequency
[i
].cell
[j
].base_station_identity_code
);
836 blobmsg_add_double(&status
, "rssi",
837 ((double)res
.data
.lte_info_neighboring_gsm
.frequency
[i
].cell
[j
].rssi
)/10);
838 if (res
.data
.lte_info_neighboring_gsm
.ue_in_idle
)
839 blobmsg_add_u32(&status
, "cell_selection_rx_level",
840 res
.data
.lte_info_neighboring_gsm
.frequency
[i
].cell
[j
].cell_selection_rx_level
);
841 blobmsg_close_table(&status
, cell
);
843 blobmsg_close_table(&status
, freq
);
845 if (res
.data
.lte_info_neighboring_gsm
.frequency_n
> 0)
846 blobmsg_close_table(&status
, c
);
848 if (res
.set
.lte_info_neighboring_wcdma
) {
849 if (res
.data
.lte_info_neighboring_wcdma
.frequency_n
> 0)
850 c
= blobmsg_open_table(&status
, "lte_info_neighboring_wcdma");
851 for (i
= 0; i
< res
.data
.lte_info_neighboring_wcdma
.frequency_n
; i
++) {
852 freq
= blobmsg_open_table(&status
, NULL
);
853 blobmsg_add_u32(&status
, "channel",
854 res
.data
.lte_info_neighboring_wcdma
.frequency
[i
].utra_absolute_rf_channel_number
);
855 if (res
.data
.lte_info_neighboring_wcdma
.ue_in_idle
) {
856 print_sel_info(res
.data
.lte_info_neighboring_wcdma
.frequency
[i
].cell_reselection_priority
,
857 res
.data
.lte_info_neighboring_wcdma
.frequency
[i
].cell_reselection_high_threshold
,
858 res
.data
.lte_info_neighboring_wcdma
.frequency
[i
].cell_reselection_low_threshold
);
860 for (j
= 0; j
< res
.data
.lte_info_neighboring_wcdma
.frequency
[i
].cell_n
; j
++) {
861 cell
= blobmsg_open_table(&status
, NULL
);
862 blobmsg_add_u32(&status
, "primary_scrambling_code",
863 res
.data
.lte_info_neighboring_wcdma
.frequency
[i
].cell
[j
].primary_scrambling_code
);
864 blobmsg_add_double(&status
, "rscp",
865 ((double)res
.data
.lte_info_neighboring_wcdma
.frequency
[i
].cell
[j
].cpich_rscp
)/10);
866 blobmsg_add_double(&status
, "ecno",
867 ((double)res
.data
.lte_info_neighboring_wcdma
.frequency
[i
].cell
[j
].cpich_ecno
)/10);
868 if (res
.data
.lte_info_neighboring_wcdma
.ue_in_idle
)
869 blobmsg_add_u32(&status
, "cell_selection_rx_level",
870 res
.data
.lte_info_neighboring_wcdma
.frequency
[i
].cell
[j
].cell_selection_rx_level
);
871 blobmsg_close_table(&status
, cell
);
873 blobmsg_close_table(&status
, freq
);
875 if (res
.data
.lte_info_neighboring_wcdma
.frequency_n
> 0)
876 blobmsg_close_table(&status
, c
);
878 if (res
.set
.umts_info_neighboring_lte
) {
879 if (res
.data
.umts_info_neighboring_lte
.frequency_n
> 0)
880 c
= blobmsg_open_table(&status
, "umts_info_neighboring_lte");
881 for (i
= 0; i
< res
.data
.umts_info_neighboring_lte
.frequency_n
; i
++) {
882 freq
= blobmsg_open_table(&status
, NULL
);
883 blobmsg_add_u32(&status
, "channel",
884 res
.data
.umts_info_neighboring_lte
.frequency
[i
].eutra_absolute_rf_channel_number
);
885 print_earfcn_info(res
.data
.umts_info_neighboring_lte
.frequency
[i
].eutra_absolute_rf_channel_number
);
886 blobmsg_add_u32(&status
, "physical_cell_id",
887 res
.data
.umts_info_neighboring_lte
.frequency
[i
].physical_cell_id
);
888 blobmsg_add_double(&status
, "rsrp",
889 (double) res
.data
.umts_info_neighboring_lte
.frequency
[i
].rsrp
);
890 blobmsg_add_double(&status
, "rsrq",
891 (double) res
.data
.umts_info_neighboring_lte
.frequency
[i
].rsrq
);
892 blobmsg_add_u32(&status
, "cell_selection_rx_level",
893 res
.data
.umts_info_neighboring_lte
.frequency
[i
].cell_selection_rx_level
);
894 blobmsg_close_table(&status
, freq
);
896 if (res
.data
.umts_info_neighboring_lte
.frequency_n
> 0)
897 blobmsg_close_table(&status
, c
);
899 blobmsg_close_table(&status
, t
);
902 static enum qmi_cmd_result
903 cmd_nas_get_cell_location_info_prepare(struct qmi_dev
*qmi
, struct qmi_request
*req
, struct qmi_msg
*msg
, char *arg
)
905 qmi_set_nas_get_cell_location_info_request(msg
);
906 return QMI_CMD_REQUEST
;
909 static enum qmi_cmd_result
910 cmd_nas_get_signal_info_prepare(struct qmi_dev
*qmi
, struct qmi_request
*req
, struct qmi_msg
*msg
, char *arg
)
912 qmi_set_nas_get_signal_info_request(msg
);
913 return QMI_CMD_REQUEST
;
917 cmd_nas_get_serving_system_cb(struct qmi_dev
*qmi
, struct qmi_request
*req
, struct qmi_msg
*msg
)
919 struct qmi_nas_get_serving_system_response res
;
920 static const char *reg_states
[] = {
921 [QMI_NAS_REGISTRATION_STATE_NOT_REGISTERED
] = "not_registered",
922 [QMI_NAS_REGISTRATION_STATE_REGISTERED
] = "registered",
923 [QMI_NAS_REGISTRATION_STATE_NOT_REGISTERED_SEARCHING
] = "searching",
924 [QMI_NAS_REGISTRATION_STATE_REGISTRATION_DENIED
] = "registering_denied",
925 [QMI_NAS_REGISTRATION_STATE_UNKNOWN
] = "unknown",
929 qmi_parse_nas_get_serving_system_response(msg
, &res
);
931 c
= blobmsg_open_table(&status
, NULL
);
932 if (res
.set
.serving_system
) {
933 int state
= res
.data
.serving_system
.registration_state
;
935 if (state
> QMI_NAS_REGISTRATION_STATE_UNKNOWN
)
936 state
= QMI_NAS_REGISTRATION_STATE_UNKNOWN
;
938 blobmsg_add_string(&status
, "registration", reg_states
[state
]);
940 if (res
.set
.current_plmn
) {
941 blobmsg_add_u32(&status
, "plmn_mcc", res
.data
.current_plmn
.mcc
);
942 blobmsg_add_u32(&status
, "plmn_mnc", res
.data
.current_plmn
.mnc
);
943 if (res
.data
.current_plmn
.description
)
944 blobmsg_add_string(&status
, "plmn_description", res
.data
.current_plmn
.description
);
947 if (res
.set
.roaming_indicator
)
948 blobmsg_add_u8(&status
, "roaming", !res
.data
.roaming_indicator
);
950 blobmsg_close_table(&status
, c
);
953 static enum qmi_cmd_result
954 cmd_nas_get_serving_system_prepare(struct qmi_dev
*qmi
, struct qmi_request
*req
, struct qmi_msg
*msg
, char *arg
)
956 qmi_set_nas_get_serving_system_request(msg
);
957 return QMI_CMD_REQUEST
;
961 cmd_nas_get_plmn_cb(struct qmi_dev
*qmi
, struct qmi_request
*req
, struct qmi_msg
*msg
)
963 struct qmi_nas_get_system_selection_preference_response res
;
964 static const char *modes
[] = {
965 [QMI_NAS_NETWORK_SELECTION_PREFERENCE_AUTOMATIC
] = "automatic",
966 [QMI_NAS_NETWORK_SELECTION_PREFERENCE_MANUAL
] = "manual",
970 qmi_parse_nas_get_system_selection_preference_response(msg
, &res
);
972 c
= blobmsg_open_table(&status
, NULL
);
973 if (res
.set
.network_selection_preference
) {
974 blobmsg_add_string(&status
, "mode", modes
[res
.data
.network_selection_preference
]);
976 if (res
.set
.manual_network_selection
) {
977 blobmsg_add_u32(&status
, "mcc", res
.data
.manual_network_selection
.mcc
);
978 blobmsg_add_u32(&status
, "mnc", res
.data
.manual_network_selection
.mnc
);
981 blobmsg_close_table(&status
, c
);
984 static enum qmi_cmd_result
985 cmd_nas_get_plmn_prepare(struct qmi_dev
*qmi
, struct qmi_request
*req
, struct qmi_msg
*msg
, char *arg
)
987 qmi_set_nas_get_system_selection_preference_request(msg
);
988 return QMI_CMD_REQUEST
;
992 cmd_nas_network_scan_cb(struct qmi_dev
*qmi
, struct qmi_request
*req
, struct qmi_msg
*msg
)
994 static struct qmi_nas_network_scan_response res
;
995 const char *network_status
[] = {
1005 const char *radio
[] = {
1006 [QMI_NAS_RADIO_INTERFACE_NONE
] = "none",
1007 [QMI_NAS_RADIO_INTERFACE_CDMA_1X
] = "cdma-1x",
1008 [QMI_NAS_RADIO_INTERFACE_CDMA_1XEVDO
] = "cdma-1x_evdo",
1009 [QMI_NAS_RADIO_INTERFACE_AMPS
] = "amps",
1010 [QMI_NAS_RADIO_INTERFACE_GSM
] = "gsm",
1011 [QMI_NAS_RADIO_INTERFACE_UMTS
] = "umts",
1012 [QMI_NAS_RADIO_INTERFACE_LTE
] = "lte",
1013 [QMI_NAS_RADIO_INTERFACE_TD_SCDMA
] = "td-scdma",
1015 void *t
, *c
, *info
, *stat
;
1018 qmi_parse_nas_network_scan_response(msg
, &res
);
1020 t
= blobmsg_open_table(&status
, NULL
);
1022 c
= blobmsg_open_array(&status
, "network_info");
1023 for (i
= 0; i
< res
.data
.network_information_n
; i
++) {
1024 info
= blobmsg_open_table(&status
, NULL
);
1025 blobmsg_add_u32(&status
, "mcc", res
.data
.network_information
[i
].mcc
);
1026 blobmsg_add_u32(&status
, "mnc", res
.data
.network_information
[i
].mnc
);
1027 if (res
.data
.network_information
[i
].description
)
1028 blobmsg_add_string(&status
, "description", res
.data
.network_information
[i
].description
);
1029 stat
= blobmsg_open_array(&status
, "status");
1030 for (j
= 0; j
< ARRAY_SIZE(network_status
); j
++) {
1031 if (!(res
.data
.network_information
[i
].network_status
& (1 << j
)))
1034 blobmsg_add_string(&status
, NULL
, network_status
[j
]);
1036 blobmsg_close_array(&status
, stat
);
1037 blobmsg_close_table(&status
, info
);
1039 blobmsg_close_array(&status
, c
);
1041 c
= blobmsg_open_array(&status
, "radio_access_technology");
1042 for (i
= 0; i
< res
.data
.radio_access_technology_n
; i
++) {
1043 const char *r
= "unknown";
1044 int r_i
= res
.data
.radio_access_technology
[i
].radio_interface
;
1046 info
= blobmsg_open_table(&status
, NULL
);
1047 blobmsg_add_u32(&status
, "mcc", res
.data
.radio_access_technology
[i
].mcc
);
1048 blobmsg_add_u32(&status
, "mnc", res
.data
.radio_access_technology
[i
].mnc
);
1049 if (r_i
>= 0 && r_i
< ARRAY_SIZE(radio
))
1052 blobmsg_add_string(&status
, "radio", r
);
1053 blobmsg_close_table(&status
, info
);
1055 blobmsg_close_array(&status
, c
);
1057 blobmsg_close_table(&status
, t
);
1060 static enum qmi_cmd_result
1061 cmd_nas_network_scan_prepare(struct qmi_dev
*qmi
, struct qmi_request
*req
, struct qmi_msg
*msg
, char *arg
)
1063 struct qmi_nas_network_scan_request sreq
= {
1064 QMI_INIT(network_type
,
1065 QMI_NAS_NETWORK_SCAN_TYPE_GSM
|
1066 QMI_NAS_NETWORK_SCAN_TYPE_UMTS
|
1067 QMI_NAS_NETWORK_SCAN_TYPE_LTE
|
1068 QMI_NAS_NETWORK_SCAN_TYPE_TD_SCDMA
),
1071 qmi_set_nas_network_scan_request(msg
, &sreq
);
1072 return QMI_CMD_REQUEST
;