uqmi: add APN profile commands
[project/uqmi.git] / commands-uim.c
index 859da68e49278e2d6c50141a6becd311bf0506d1..ff01d7b4a6204bb38da0d7286957831bb8c73555 100644 (file)
  * Boston, MA 02110-1301 USA.
  */
 
+static int uim_slot = 0;
+
 #define cmd_uim_verify_pin1_cb no_cb
 static enum qmi_cmd_result
 cmd_uim_verify_pin1_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
 {
        struct qmi_uim_verify_pin_request data = {
-               QMI_INIT_SEQUENCE(session_information,
+               QMI_INIT_SEQUENCE(session,
                        .session_type = QMI_UIM_SESSION_TYPE_CARD_SLOT_1,
-                       ""
+                       .application_identifier_n = 0
                ),
                QMI_INIT_SEQUENCE(info,
                        .pin_id = QMI_UIM_PIN_ID_PIN1,
@@ -42,9 +44,9 @@ static enum qmi_cmd_result
 cmd_uim_verify_pin2_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
 {
        struct qmi_uim_verify_pin_request data = {
-               QMI_INIT_SEQUENCE(session_information,
+               QMI_INIT_SEQUENCE(session,
                        .session_type = QMI_UIM_SESSION_TYPE_CARD_SLOT_1,
-                       ""
+                       .application_identifier_n = 0
                ),
                QMI_INIT_SEQUENCE(info,
                        .pin_id = QMI_UIM_PIN_ID_PIN2,
@@ -54,3 +56,129 @@ cmd_uim_verify_pin2_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct
        qmi_set_uim_verify_pin_request(msg, &data);
        return QMI_CMD_REQUEST;
 }
+
+static void cmd_uim_get_sim_state_cb(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg)
+{
+       struct qmi_uim_get_card_status_response res;
+       void * const card_table = blobmsg_open_table(&status, NULL);
+       static const char *card_application_states[] = {
+               [QMI_UIM_CARD_APPLICATION_STATE_UNKNOWN] = "unknown",
+               [QMI_UIM_CARD_APPLICATION_STATE_DETECTED] = "detected",
+               [QMI_UIM_CARD_APPLICATION_STATE_PIN1_OR_UPIN_PIN_REQUIRED] = "pin-required",
+               [QMI_UIM_CARD_APPLICATION_STATE_PUK1_OR_UPIN_PUK_REQUIRED] = "puk-required",
+               [QMI_UIM_CARD_APPLICATION_STATE_CHECK_PERSONALIZATION_STATE] = "check-personalization-state",
+               [QMI_UIM_CARD_APPLICATION_STATE_PIN1_BLOCKED] = "pin1-blocked",
+               [QMI_UIM_CARD_APPLICATION_STATE_ILLEGAL] = "illegal",
+               [QMI_UIM_CARD_APPLICATION_STATE_READY] = "ready",
+       };
+
+       qmi_parse_uim_get_card_status_response(msg, &res);
+
+       for (int i = 0; i < res.data.card_status.cards_n; ++i) {
+               if (res.data.card_status.cards[i].card_state != QMI_UIM_CARD_STATE_PRESENT)
+                       continue;
+
+               uint8_t card_application_state;
+               uint8_t pin1_state = res.data.card_status.cards[i].upin_state;
+               uint8_t pin1_retries = res.data.card_status.cards[i].upin_retries;
+               uint8_t puk1_retries = res.data.card_status.cards[i].upuk_retries;
+               uint8_t pin2_state;
+               uint8_t pin2_retries;
+               uint8_t puk2_retries;
+               bool has_pin2 = false;
+
+               for (int j = 0; j < res.data.card_status.cards[i].applications_n; ++j) {
+                       if (res.data.card_status.cards[i].applications[j].type == QMI_UIM_CARD_APPLICATION_TYPE_UNKNOWN)
+                               continue;
+
+                       card_application_state = pin1_state = res.data.card_status.cards[i].applications[j].state;
+
+                       if (!res.data.card_status.cards[i].applications[j].upin_replaces_pin1) {
+                               pin1_state = res.data.card_status.cards[i].applications[j].pin1_state;
+                               pin1_retries = res.data.card_status.cards[i].applications[j].pin1_retries;
+                               puk1_retries = res.data.card_status.cards[i].applications[j].puk1_retries;
+                       }
+
+                       pin2_state = res.data.card_status.cards[i].applications[j].pin2_state;
+                       pin2_retries = res.data.card_status.cards[i].applications[j].pin2_retries;
+                       puk2_retries = res.data.card_status.cards[i].applications[j].puk2_retries;
+                       has_pin2 = true;
+
+                       break; /* handle first application only for now */
+               }
+
+               if (card_application_state > QMI_UIM_CARD_APPLICATION_STATE_READY)
+                       card_application_state = QMI_UIM_CARD_APPLICATION_STATE_UNKNOWN;
+
+               blobmsg_add_string(&status, "card_application_state", card_application_states[card_application_state]);
+               blobmsg_add_string(&status, "pin1_status", get_pin_status(pin1_state));
+               blobmsg_add_u32(&status, "pin1_verify_tries", pin1_retries);
+               blobmsg_add_u32(&status, "pin1_unlock_tries", puk1_retries);
+               if (has_pin2) {
+                       blobmsg_add_string(&status, "pin2_status", get_pin_status(pin2_state));
+                       blobmsg_add_u32(&status, "pin2_verify_tries", pin2_retries);
+                       blobmsg_add_u32(&status, "pin2_unlock_tries", puk2_retries);
+               }
+
+               break; /* handle only first preset SIM card for now */
+       }
+
+       blobmsg_close_table(&status, card_table);
+}
+
+static enum qmi_cmd_result
+cmd_uim_get_sim_state_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
+{
+       qmi_set_uim_get_card_status_request(msg);
+       return QMI_CMD_REQUEST;
+}
+
+#define cmd_uim_slot_cb no_cb
+static enum qmi_cmd_result
+cmd_uim_slot_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
+{
+       char *err;
+       int value = strtoul(arg, &err, 10);
+       if ((err && *err) || value < 1 || value > 2) {
+               uqmi_add_error("Invalid UIM-Slot value. Allowed: [1,2]");
+               return QMI_CMD_EXIT;
+       }
+
+       uim_slot = value;
+
+       return QMI_CMD_DONE;
+}
+
+#define cmd_uim_power_off_cb no_cb
+static enum qmi_cmd_result
+cmd_uim_power_off_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
+{
+       struct qmi_uim_power_off_sim_request data = {
+               QMI_INIT(slot, uim_slot)
+       };
+
+       if (!uim_slot) {
+               uqmi_add_error("UIM-Slot not set. Use --uim-slot <slot> to set it.");
+               return QMI_CMD_EXIT;
+       }
+
+       qmi_set_uim_power_off_sim_request(msg, &data);
+       return QMI_CMD_REQUEST;
+}
+
+#define cmd_uim_power_on_cb no_cb
+static enum qmi_cmd_result
+cmd_uim_power_on_prepare(struct qmi_dev *qmi, struct qmi_request *req, struct qmi_msg *msg, char *arg)
+{
+       struct qmi_uim_power_on_sim_request data = {
+               QMI_INIT(slot, uim_slot)
+       };
+
+       if (!uim_slot) {
+               uqmi_add_error("UIM-Slot not set. Use --uim-slot <slot> to set it.");
+               return QMI_CMD_EXIT;
+       }
+
+       qmi_set_uim_power_on_sim_request(msg, &data);
+       return QMI_CMD_REQUEST;
+}