commands-uim.c 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115
  1. /*
  2. * uqmi -- tiny QMI support implementation
  3. *
  4. * Copyright (C) 2016 Felix Fietkau <nbd@nbd.name>
  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. #define cmd_uim_verify_pin1_cb no_cb
  22. static enum qmi_cmd_result
  23. cmd_uim_verify_pin1_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
  24. {
  25. struct qmi_uim_verify_pin_request data = {
  26. QMI_INIT_SEQUENCE(session,
  27. .session_type = QMI_UIM_SESSION_TYPE_CARD_SLOT_1,
  28. .application_identifier_n = 0
  29. ),
  30. QMI_INIT_SEQUENCE(info,
  31. .pin_id = QMI_UIM_PIN_ID_PIN1,
  32. .pin_value = arg
  33. )
  34. };
  35. qmi_set_uim_verify_pin_request(msg, &data);
  36. return QMI_CMD_REQUEST;
  37. }
  38. #define cmd_uim_verify_pin2_cb no_cb
  39. static enum qmi_cmd_result
  40. cmd_uim_verify_pin2_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
  41. {
  42. struct qmi_uim_verify_pin_request data = {
  43. QMI_INIT_SEQUENCE(session,
  44. .session_type = QMI_UIM_SESSION_TYPE_CARD_SLOT_1,
  45. .application_identifier_n = 0
  46. ),
  47. QMI_INIT_SEQUENCE(info,
  48. .pin_id = QMI_UIM_PIN_ID_PIN2,
  49. .pin_value = arg
  50. )
  51. };
  52. qmi_set_uim_verify_pin_request(msg, &data);
  53. return QMI_CMD_REQUEST;
  54. }
  55. static void cmd_uim_get_sim_state_cb(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg)
  56. {
  57. struct qmi_uim_get_card_status_response res;
  58. void * const card_table = blobmsg_open_table(&status, NULL);
  59. qmi_parse_uim_get_card_status_response(msg, &res);
  60. for (int i = 0; i < res.data.card_status.cards_n; ++i) {
  61. if (res.data.card_status.cards[i].card_state != QMI_UIM_CARD_STATE_PRESENT)
  62. continue;
  63. uint8_t pin1_state = res.data.card_status.cards[i].upin_state;
  64. uint8_t pin1_retries = res.data.card_status.cards[i].upin_retries;
  65. uint8_t puk1_retries = res.data.card_status.cards[i].upuk_retries;
  66. uint8_t pin2_state;
  67. uint8_t pin2_retries;
  68. uint8_t puk2_retries;
  69. bool has_pin2 = false;
  70. for (int j = 0; j < res.data.card_status.cards[i].applications_n; ++j) {
  71. if (res.data.card_status.cards[i].applications[j].type == QMI_UIM_CARD_APPLICATION_TYPE_UNKNOWN)
  72. continue;
  73. if (!res.data.card_status.cards[i].applications[j].upin_replaces_pin1) {
  74. pin1_state = res.data.card_status.cards[i].applications[j].pin1_state;
  75. pin1_retries = res.data.card_status.cards[i].applications[j].pin1_retries;
  76. puk1_retries = res.data.card_status.cards[i].applications[j].puk1_retries;
  77. }
  78. pin2_state = res.data.card_status.cards[i].applications[j].pin2_state;
  79. pin2_retries = res.data.card_status.cards[i].applications[j].pin2_retries;
  80. puk2_retries = res.data.card_status.cards[i].applications[j].puk2_retries;
  81. has_pin2 = true;
  82. break; /* handle first application only for now */
  83. }
  84. blobmsg_add_string(&status, "pin1_status", get_pin_status(pin1_state));
  85. blobmsg_add_u32(&status, "pin1_verify_tries", pin1_retries);
  86. blobmsg_add_u32(&status, "pin1_unlock_tries", puk1_retries);
  87. if (has_pin2) {
  88. blobmsg_add_string(&status, "pin2_status", get_pin_status(pin2_state));
  89. blobmsg_add_u32(&status, "pin2_verify_tries", pin2_retries);
  90. blobmsg_add_u32(&status, "pin2_unlock_tries", puk2_retries);
  91. }
  92. break; /* handle only first preset SIM card for now */
  93. }
  94. blobmsg_close_table(&status, card_table);
  95. }
  96. static enum qmi_cmd_result
  97. cmd_uim_get_sim_state_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
  98. {
  99. qmi_set_uim_get_card_status_request(msg);
  100. return QMI_CMD_REQUEST;
  101. }