commands-dms.c 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418
  1. /*
  2. * uqmi -- tiny QMI support implementation
  3. *
  4. * Copyright (C) 2014-2015 Felix Fietkau <nbd@openwrt.org>
  5. *
  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.
  10. *
  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.
  15. *
  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.
  20. */
  21. #include "qmi-message.h"
  22. static struct {
  23. QmiDmsUimPinId pin_id;
  24. char* pin;
  25. char* new_pin;
  26. char* puk;
  27. } dms_req_data;
  28. static void cmd_dms_get_capabilities_cb(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg)
  29. {
  30. void *t, *networks;
  31. int i;
  32. struct qmi_dms_get_capabilities_response res;
  33. const char *radio_cap[] = {
  34. [QMI_DMS_RADIO_INTERFACE_CDMA20001X] = "cdma1x",
  35. [QMI_DMS_RADIO_INTERFACE_EVDO] = "cdma1xevdo",
  36. [QMI_DMS_RADIO_INTERFACE_GSM] = "gsm",
  37. [QMI_DMS_RADIO_INTERFACE_UMTS] = "umts",
  38. [QMI_DMS_RADIO_INTERFACE_LTE] = "lte",
  39. };
  40. const char *service_cap[] = {
  41. [QMI_DMS_DATA_SERVICE_CAPABILITY_NONE] = "none",
  42. [QMI_DMS_DATA_SERVICE_CAPABILITY_CS] = "cs",
  43. [QMI_DMS_DATA_SERVICE_CAPABILITY_PS] = "ps",
  44. [QMI_DMS_DATA_SERVICE_CAPABILITY_SIMULTANEOUS_CS_PS] = "simultaneous_cs_ps",
  45. [QMI_DMS_DATA_SERVICE_CAPABILITY_NON_SIMULTANEOUS_CS_PS] = "non_simultaneous_cs_ps",
  46. };
  47. qmi_parse_dms_get_capabilities_response(msg, &res);
  48. t = blobmsg_open_table(&status, NULL);
  49. blobmsg_add_u32(&status, "max_tx_channel_rate", (int32_t) res.data.info.max_tx_channel_rate);
  50. blobmsg_add_u32(&status, "max_rx_channel_rate", (int32_t) res.data.info.max_rx_channel_rate);
  51. if ((int)res.data.info.data_service_capability >= 0 && res.data.info.data_service_capability < ARRAY_SIZE(service_cap))
  52. blobmsg_add_string(&status, "data_service", service_cap[res.data.info.data_service_capability]);
  53. if(res.data.info.sim_capability == QMI_DMS_SIM_CAPABILITY_NOT_SUPPORTED)
  54. blobmsg_add_string(&status, "sim", "not supported");
  55. else if(res.data.info.sim_capability == QMI_DMS_SIM_CAPABILITY_SUPPORTED)
  56. blobmsg_add_string(&status, "sim", "supported");
  57. networks = blobmsg_open_array(&status, "networks");
  58. for (i = 0; i < res.data.info.radio_interface_list_n; i++) {
  59. if ((int)res.data.info.radio_interface_list[i] >= 0 && res.data.info.radio_interface_list[i] < ARRAY_SIZE(radio_cap))
  60. blobmsg_add_string(&status, NULL, radio_cap[res.data.info.radio_interface_list[i]]);
  61. else
  62. blobmsg_add_string(&status, NULL, "unknown");
  63. }
  64. blobmsg_close_array(&status, networks);
  65. blobmsg_close_table(&status, t);
  66. }
  67. static enum qmi_cmd_result
  68. cmd_dms_get_capabilities_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
  69. {
  70. qmi_set_dms_get_capabilities_request(msg);
  71. return QMI_CMD_REQUEST;
  72. }
  73. const char *get_pin_status(int status)
  74. {
  75. static const char *pin_status[] = {
  76. [QMI_DMS_UIM_PIN_STATUS_NOT_INITIALIZED] = "not_initialized",
  77. [QMI_DMS_UIM_PIN_STATUS_ENABLED_NOT_VERIFIED] = "not_verified",
  78. [QMI_DMS_UIM_PIN_STATUS_ENABLED_VERIFIED] = "verified",
  79. [QMI_DMS_UIM_PIN_STATUS_DISABLED] = "disabled",
  80. [QMI_DMS_UIM_PIN_STATUS_BLOCKED] = "blocked",
  81. [QMI_DMS_UIM_PIN_STATUS_PERMANENTLY_BLOCKED] = "permanently_blocked",
  82. [QMI_DMS_UIM_PIN_STATUS_UNBLOCKED] = "unblocked",
  83. [QMI_DMS_UIM_PIN_STATUS_CHANGED] = "changed",
  84. };
  85. const char *res = "Unknown";
  86. if (status < ARRAY_SIZE(pin_status) && pin_status[status])
  87. res = pin_status[status];
  88. return res;
  89. }
  90. static void cmd_dms_get_pin_status_cb(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg)
  91. {
  92. struct qmi_dms_uim_get_pin_status_response res;
  93. void *c;
  94. qmi_parse_dms_uim_get_pin_status_response(msg, &res);
  95. c = blobmsg_open_table(&status, NULL);
  96. if (res.set.pin1_status) {
  97. blobmsg_add_string(&status, "pin1_status", get_pin_status(res.data.pin1_status.current_status));
  98. blobmsg_add_u32(&status, "pin1_verify_tries", (int32_t) res.data.pin1_status.verify_retries_left);
  99. blobmsg_add_u32(&status, "pin1_unblock_tries", (int32_t) res.data.pin1_status.unblock_retries_left);
  100. }
  101. if (res.set.pin2_status) {
  102. blobmsg_add_string(&status, "pin2_status", get_pin_status(res.data.pin2_status.current_status));
  103. blobmsg_add_u32(&status, "pin2_verify_tries", (int32_t) res.data.pin2_status.verify_retries_left);
  104. blobmsg_add_u32(&status, "pin2_unblock_tries", (int32_t) res.data.pin2_status.unblock_retries_left);
  105. }
  106. blobmsg_close_table(&status, c);
  107. }
  108. static enum qmi_cmd_result
  109. cmd_dms_get_pin_status_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
  110. {
  111. qmi_set_dms_uim_get_pin_status_request(msg);
  112. return QMI_CMD_REQUEST;
  113. }
  114. #define cmd_dms_verify_pin1_cb no_cb
  115. static enum qmi_cmd_result
  116. cmd_dms_verify_pin1_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
  117. {
  118. struct qmi_dms_uim_verify_pin_request data = {
  119. QMI_INIT_SEQUENCE(info,
  120. .pin_id = QMI_DMS_UIM_PIN_ID_PIN,
  121. .pin = arg
  122. )
  123. };
  124. qmi_set_dms_uim_verify_pin_request(msg, &data);
  125. return QMI_CMD_REQUEST;
  126. }
  127. #define cmd_dms_verify_pin2_cb no_cb
  128. static enum qmi_cmd_result
  129. cmd_dms_verify_pin2_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
  130. {
  131. struct qmi_dms_uim_verify_pin_request data = {
  132. QMI_INIT_SEQUENCE(info,
  133. .pin_id = QMI_DMS_UIM_PIN_ID_PIN2,
  134. .pin = arg
  135. )
  136. };
  137. qmi_set_dms_uim_verify_pin_request(msg, &data);
  138. return QMI_CMD_REQUEST;
  139. }
  140. #define cmd_dms_set_pin_cb no_cb
  141. static enum qmi_cmd_result
  142. cmd_dms_set_pin_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
  143. {
  144. dms_req_data.pin = arg;
  145. return QMI_CMD_DONE;
  146. }
  147. static enum qmi_cmd_result
  148. cmd_dms_set_pin_protection_prepare(struct qmi_msg *msg, char *arg)
  149. {
  150. if (!dms_req_data.pin) {
  151. uqmi_add_error("Missing argument");
  152. return QMI_CMD_EXIT;
  153. }
  154. int is_enabled;
  155. if (strcasecmp(arg, "disabled") == 0)
  156. is_enabled = false;
  157. else if (strcasecmp(arg, "enabled") == 0)
  158. is_enabled = true;
  159. else {
  160. uqmi_add_error("Invalid value (valid: disabled, enabled)");
  161. return QMI_CMD_EXIT;
  162. }
  163. struct qmi_dms_uim_set_pin_protection_request dms_pin_protection_req = {
  164. QMI_INIT_SEQUENCE(info,
  165. .pin_id = dms_req_data.pin_id
  166. ),
  167. QMI_INIT_PTR(info.pin, dms_req_data.pin),
  168. QMI_INIT_PTR(info.protection_enabled, is_enabled)
  169. };
  170. qmi_set_dms_uim_set_pin_protection_request(msg, &dms_pin_protection_req);
  171. return QMI_CMD_REQUEST;
  172. }
  173. #define cmd_dms_set_pin1_protection_cb no_cb
  174. static enum qmi_cmd_result
  175. cmd_dms_set_pin1_protection_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
  176. {
  177. dms_req_data.pin_id = QMI_DMS_UIM_PIN_ID_PIN;
  178. return cmd_dms_set_pin_protection_prepare(msg, arg);
  179. }
  180. #define cmd_dms_set_pin2_protection_cb no_cb
  181. static enum qmi_cmd_result
  182. cmd_dms_set_pin2_protection_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
  183. {
  184. dms_req_data.pin_id = QMI_DMS_UIM_PIN_ID_PIN2;
  185. return cmd_dms_set_pin_protection_prepare(msg, arg);
  186. }
  187. static enum qmi_cmd_result
  188. cmd_dms_change_pin_prepare(struct qmi_msg *msg, char *arg)
  189. {
  190. if (!dms_req_data.pin || !dms_req_data.new_pin) {
  191. uqmi_add_error("Missing argument");
  192. return QMI_CMD_EXIT;
  193. }
  194. struct qmi_dms_uim_change_pin_request dms_change_pin_req = {
  195. QMI_INIT_SEQUENCE(info,
  196. .pin_id = dms_req_data.pin_id
  197. ),
  198. QMI_INIT_PTR(info.old_pin, dms_req_data.pin),
  199. QMI_INIT_PTR(info.new_pin, dms_req_data.new_pin)
  200. };
  201. qmi_set_dms_uim_change_pin_request(msg, &dms_change_pin_req);
  202. return QMI_CMD_REQUEST;
  203. }
  204. #define cmd_dms_change_pin1_cb no_cb
  205. static enum qmi_cmd_result
  206. cmd_dms_change_pin1_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
  207. {
  208. dms_req_data.pin_id = QMI_DMS_UIM_PIN_ID_PIN;
  209. return cmd_dms_change_pin_prepare(msg, arg);
  210. }
  211. #define cmd_dms_change_pin2_cb no_cb
  212. static enum qmi_cmd_result
  213. cmd_dms_change_pin2_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
  214. {
  215. dms_req_data.pin_id = QMI_DMS_UIM_PIN_ID_PIN2;
  216. return cmd_dms_change_pin_prepare(msg, arg);
  217. }
  218. #define cmd_dms_set_new_pin_cb no_cb
  219. static enum qmi_cmd_result
  220. cmd_dms_set_new_pin_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
  221. {
  222. dms_req_data.new_pin = arg;
  223. return QMI_CMD_DONE;
  224. }
  225. #define cmd_dms_set_puk_cb no_cb
  226. static enum qmi_cmd_result
  227. cmd_dms_set_puk_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
  228. {
  229. dms_req_data.puk = arg;
  230. return QMI_CMD_DONE;
  231. }
  232. #define cmd_dms_unblock_pin1_cb no_cb
  233. static enum qmi_cmd_result
  234. cmd_dms_unblock_pin1_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
  235. {
  236. if (!dms_req_data.puk || !dms_req_data.new_pin) {
  237. uqmi_add_error("Missing argument");
  238. return QMI_CMD_EXIT;
  239. }
  240. struct qmi_dms_uim_unblock_pin_request dms_unlock_pin_req = {
  241. QMI_INIT_SEQUENCE(info,
  242. .pin_id = QMI_DMS_UIM_PIN_ID_PIN
  243. ),
  244. QMI_INIT_PTR(info.puk, dms_req_data.puk),
  245. QMI_INIT_PTR(info.new_pin, dms_req_data.new_pin)
  246. };
  247. qmi_set_dms_uim_unblock_pin_request(msg, &dms_unlock_pin_req);
  248. return QMI_CMD_REQUEST;
  249. }
  250. #define cmd_dms_unblock_pin2_cb no_cb
  251. static enum qmi_cmd_result
  252. cmd_dms_unblock_pin2_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
  253. {
  254. if (!dms_req_data.puk || !dms_req_data.new_pin) {
  255. uqmi_add_error("Missing argument");
  256. return QMI_CMD_EXIT;
  257. }
  258. struct qmi_dms_uim_unblock_pin_request dms_unlock_pin_req = {
  259. QMI_INIT_SEQUENCE(info,
  260. .pin_id = QMI_DMS_UIM_PIN_ID_PIN2
  261. ),
  262. QMI_INIT_PTR(info.puk, dms_req_data.puk),
  263. QMI_INIT_PTR(info.new_pin, dms_req_data.new_pin)
  264. };
  265. qmi_set_dms_uim_unblock_pin_request(msg, &dms_unlock_pin_req);
  266. return QMI_CMD_REQUEST;
  267. }
  268. static void cmd_dms_get_iccid_cb(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg)
  269. {
  270. struct qmi_dms_uim_get_iccid_response res;
  271. qmi_parse_dms_uim_get_iccid_response(msg, &res);
  272. if (res.data.iccid)
  273. blobmsg_add_string(&status, NULL, res.data.iccid);
  274. }
  275. static enum qmi_cmd_result
  276. cmd_dms_get_iccid_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
  277. {
  278. qmi_set_dms_uim_get_iccid_request(msg);
  279. return QMI_CMD_REQUEST;
  280. }
  281. static void cmd_dms_get_imsi_cb(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg)
  282. {
  283. struct qmi_dms_uim_get_imsi_response res;
  284. qmi_parse_dms_uim_get_imsi_response(msg, &res);
  285. if (res.data.imsi)
  286. blobmsg_add_string(&status, NULL, res.data.imsi);
  287. }
  288. static enum qmi_cmd_result
  289. cmd_dms_get_imsi_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
  290. {
  291. qmi_set_dms_uim_get_imsi_request(msg);
  292. return QMI_CMD_REQUEST;
  293. }
  294. static void cmd_dms_get_msisdn_cb(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg)
  295. {
  296. struct qmi_dms_get_msisdn_response res;
  297. qmi_parse_dms_get_msisdn_response(msg, &res);
  298. if (res.data.msisdn)
  299. blobmsg_add_string(&status, NULL, res.data.msisdn);
  300. }
  301. static enum qmi_cmd_result
  302. cmd_dms_get_msisdn_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
  303. {
  304. qmi_set_dms_get_msisdn_request(msg);
  305. return QMI_CMD_REQUEST;
  306. }
  307. static void cmd_dms_get_imei_cb(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg)
  308. {
  309. struct qmi_dms_get_ids_response res;
  310. qmi_parse_dms_get_ids_response(msg, &res);
  311. if (res.data.imei)
  312. blobmsg_add_string(&status, NULL, res.data.imei);
  313. }
  314. static enum qmi_cmd_result
  315. cmd_dms_get_imei_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
  316. {
  317. qmi_set_dms_get_ids_request(msg);
  318. return QMI_CMD_REQUEST;
  319. }
  320. #define cmd_dms_reset_cb no_cb
  321. static enum qmi_cmd_result
  322. cmd_dms_reset_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
  323. {
  324. qmi_set_dms_reset_request(msg);
  325. return QMI_CMD_REQUEST;
  326. }
  327. #define cmd_dms_set_operating_mode_cb no_cb
  328. static enum qmi_cmd_result
  329. cmd_dms_set_operating_mode_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
  330. {
  331. static const char *modes[] = {
  332. [QMI_DMS_OPERATING_MODE_ONLINE] = "online",
  333. [QMI_DMS_OPERATING_MODE_LOW_POWER] = "low_power",
  334. [QMI_DMS_OPERATING_MODE_FACTORY_TEST] = "factory_test",
  335. [QMI_DMS_OPERATING_MODE_OFFLINE] = "offline",
  336. [QMI_DMS_OPERATING_MODE_RESET] = "reset",
  337. [QMI_DMS_OPERATING_MODE_SHUTTING_DOWN] = "shutting_down",
  338. [QMI_DMS_OPERATING_MODE_PERSISTENT_LOW_POWER] = "persistent_low_power",
  339. [QMI_DMS_OPERATING_MODE_MODE_ONLY_LOW_POWER] = "mode_only_low_power",
  340. };
  341. static struct qmi_dms_set_operating_mode_request sreq = {
  342. QMI_INIT(mode, QMI_DMS_OPERATING_MODE_ONLINE),
  343. };
  344. int i;
  345. for (i = 0; i < ARRAY_SIZE(modes); i++) {
  346. if (!modes[i])
  347. continue;
  348. if (strcmp(arg, modes[i]) != 0)
  349. continue;
  350. sreq.data.mode = i;
  351. qmi_set_dms_set_operating_mode_request(msg, &sreq);
  352. return QMI_CMD_REQUEST;
  353. }
  354. return uqmi_add_error("Invalid argument");
  355. }
  356. #define cmd_dms_set_fcc_authentication_cb no_cb
  357. static enum qmi_cmd_result
  358. cmd_dms_set_fcc_authentication_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
  359. {
  360. qmi_set_dms_set_fcc_authentication_request(msg);
  361. return QMI_CMD_REQUEST;
  362. }