Harald Welte | 3aa901d | 2018-08-13 18:32:36 +0200 | [diff] [blame] | 1 | |
| 2 | |
| 3 | #include <asn1c/asn_application.h> |
| 4 | #include <asn1c/der_encoder.h> |
| 5 | |
| 6 | #include <osmocom/core/msgb.h> |
| 7 | #include <osmocom/rspro/RsproPDU.h> |
| 8 | |
Harald Welte | 137c440 | 2018-08-17 21:25:56 +0200 | [diff] [blame] | 9 | #include "rspro_util.h" |
| 10 | |
Harald Welte | 57555aa | 2018-08-17 22:05:06 +0200 | [diff] [blame] | 11 | #define ASN_ALLOC_COPY(out, in) \ |
| 12 | do { \ |
| 13 | if (in) { \ |
| 14 | out = CALLOC(1, sizeof(*in)); \ |
| 15 | OSMO_ASSERT(out); \ |
| 16 | memcpy(out, in, sizeof(*in)); \ |
| 17 | } \ |
| 18 | } while (0) |
| 19 | |
| 20 | |
Harald Welte | 3aa901d | 2018-08-13 18:32:36 +0200 | [diff] [blame] | 21 | struct msgb *rspro_msgb_alloc(void) |
| 22 | { |
| 23 | return msgb_alloc_headroom(1024, 8, "RSPRO"); |
| 24 | } |
| 25 | |
| 26 | /*! BER-Encode an RSPRO message into msgb. |
| 27 | * \param[in] pdu Structure describing RSPRO PDU. Is freed by this function on success |
| 28 | * \returns callee-allocated message buffer containing encoded RSPRO PDU; NULL on error. |
| 29 | */ |
| 30 | struct msgb *rspro_enc_msg(RsproPDU_t *pdu) |
| 31 | { |
| 32 | struct msgb *msg = rspro_msgb_alloc(); |
| 33 | asn_enc_rval_t rval; |
| 34 | |
| 35 | if (!msg) |
| 36 | return NULL; |
| 37 | |
Harald Welte | fd47119 | 2018-09-24 14:51:14 +0200 | [diff] [blame] | 38 | msg->l2h = msg->data; |
Harald Welte | 6b8d4f8 | 2018-08-17 22:06:24 +0200 | [diff] [blame] | 39 | rval = der_encode_to_buffer(&asn_DEF_RsproPDU, pdu, msgb_data(msg), msgb_tailroom(msg)); |
Harald Welte | 3aa901d | 2018-08-13 18:32:36 +0200 | [diff] [blame] | 40 | if (rval.encoded < 0) { |
Harald Welte | a2b23c3 | 2018-08-17 22:09:06 +0200 | [diff] [blame] | 41 | fprintf(stderr, "Failed to encode %s\n", rval.failed_type->name); |
Harald Welte | 703d686 | 2018-09-24 17:44:50 +0200 | [diff] [blame] | 42 | msgb_free(msg); |
Harald Welte | 3aa901d | 2018-08-13 18:32:36 +0200 | [diff] [blame] | 43 | return NULL; |
| 44 | } |
Harald Welte | d5c5c0b | 2018-08-17 22:04:01 +0200 | [diff] [blame] | 45 | msgb_put(msg, rval.encoded); |
Harald Welte | 3aa901d | 2018-08-13 18:32:36 +0200 | [diff] [blame] | 46 | |
| 47 | ASN_STRUCT_FREE(asn_DEF_RsproPDU, pdu); |
| 48 | |
| 49 | return msg; |
| 50 | } |
| 51 | |
| 52 | /* consumes 'msg' _if_ it is successful */ |
| 53 | RsproPDU_t *rspro_dec_msg(struct msgb *msg) |
| 54 | { |
Harald Welte | 9ebbacc | 2018-09-24 17:43:39 +0200 | [diff] [blame] | 55 | RsproPDU_t *pdu = NULL; |
Harald Welte | 3aa901d | 2018-08-13 18:32:36 +0200 | [diff] [blame] | 56 | asn_dec_rval_t rval; |
| 57 | |
Harald Welte | 7585286 | 2018-09-24 14:55:34 +0200 | [diff] [blame] | 58 | printf("decoding %s\n", msgb_hexdump(msg)); |
| 59 | rval = ber_decode(NULL, &asn_DEF_RsproPDU, (void **) &pdu, msgb_l2(msg), msgb_l2len(msg)); |
Harald Welte | 3aa901d | 2018-08-13 18:32:36 +0200 | [diff] [blame] | 60 | if (rval.code != RC_OK) { |
Harald Welte | 7585286 | 2018-09-24 14:55:34 +0200 | [diff] [blame] | 61 | fprintf(stderr, "Failed to decode: %d. Consumed %lu of %u bytes\n", |
| 62 | rval.code, rval.consumed, msgb_length(msg)); |
| 63 | msgb_free(msg); |
Harald Welte | 3aa901d | 2018-08-13 18:32:36 +0200 | [diff] [blame] | 64 | return NULL; |
| 65 | } |
| 66 | |
| 67 | msgb_free(msg); |
| 68 | |
| 69 | return pdu; |
| 70 | } |
| 71 | |
Harald Welte | 3aa901d | 2018-08-13 18:32:36 +0200 | [diff] [blame] | 72 | static void fill_comp_id(ComponentIdentity_t *out, const struct app_comp_id *in) |
| 73 | { |
Harald Welte | 137c440 | 2018-08-17 21:25:56 +0200 | [diff] [blame] | 74 | out->type = in->type; |
Harald Welte | 3aa901d | 2018-08-13 18:32:36 +0200 | [diff] [blame] | 75 | OCTET_STRING_fromString(&out->name, in->name); |
| 76 | OCTET_STRING_fromString(&out->software, in->software); |
| 77 | OCTET_STRING_fromString(&out->swVersion, in->sw_version); |
| 78 | if (strlen(in->hw_manufacturer)) |
| 79 | out->hwManufacturer = OCTET_STRING_new_fromBuf(&asn_DEF_ComponentName, |
| 80 | in->hw_manufacturer, -1); |
| 81 | if (strlen(in->hw_model)) |
| 82 | out->hwModel = OCTET_STRING_new_fromBuf(&asn_DEF_ComponentName, in->hw_model, -1); |
| 83 | if (strlen(in->hw_serial_nr)) |
| 84 | out->hwSerialNr = OCTET_STRING_new_fromBuf(&asn_DEF_ComponentName, in->hw_serial_nr, -1); |
| 85 | if (strlen(in->hw_version)) |
| 86 | out->hwVersion = OCTET_STRING_new_fromBuf(&asn_DEF_ComponentName, in->hw_version, -1); |
| 87 | if (strlen(in->fw_version)) |
| 88 | out->fwVersion = OCTET_STRING_new_fromBuf(&asn_DEF_ComponentName, in->fw_version, -1); |
| 89 | } |
| 90 | |
| 91 | static void fill_ip4_port(IpPort_t *out, uint32_t ip, uint16_t port) |
| 92 | { |
| 93 | uint32_t ip_n = htonl(ip); |
| 94 | out->ip.present = IpAddress_PR_ipv4; |
| 95 | OCTET_STRING_fromBuf(&out->ip.choice.ipv4, (const char *) &ip_n, 4); |
| 96 | out->port = htons(port); |
| 97 | } |
| 98 | |
| 99 | |
| 100 | RsproPDU_t *rspro_gen_ConnectBankReq(const struct app_comp_id *a_cid, |
| 101 | uint16_t bank_id, uint16_t num_slots) |
| 102 | { |
| 103 | RsproPDU_t *pdu = CALLOC(1, sizeof(*pdu)); |
| 104 | if (!pdu) |
| 105 | return NULL; |
| 106 | pdu->msg.present = RsproPDUchoice_PR_connectBankReq; |
| 107 | fill_comp_id(&pdu->msg.choice.connectBankReq.identity, a_cid); |
| 108 | pdu->msg.choice.connectBankReq.bankId = bank_id; |
| 109 | pdu->msg.choice.connectBankReq.numberOfSlots = num_slots; |
| 110 | |
| 111 | return pdu; |
| 112 | } |
| 113 | |
Harald Welte | 57555aa | 2018-08-17 22:05:06 +0200 | [diff] [blame] | 114 | RsproPDU_t *rspro_gen_ConnectClientReq(const struct app_comp_id *a_cid, const ClientSlot_t *client) |
Harald Welte | 3aa901d | 2018-08-13 18:32:36 +0200 | [diff] [blame] | 115 | { |
| 116 | RsproPDU_t *pdu = CALLOC(1, sizeof(*pdu)); |
| 117 | if (!pdu) |
| 118 | return NULL; |
| 119 | pdu->msg.present = RsproPDUchoice_PR_connectClientReq; |
| 120 | fill_comp_id(&pdu->msg.choice.connectClientReq.identity, a_cid); |
Harald Welte | 57555aa | 2018-08-17 22:05:06 +0200 | [diff] [blame] | 121 | ASN_ALLOC_COPY(pdu->msg.choice.connectClientReq.clientSlot, client); |
Harald Welte | 3aa901d | 2018-08-13 18:32:36 +0200 | [diff] [blame] | 122 | |
| 123 | return pdu; |
| 124 | } |
| 125 | |
Harald Welte | 10f6c21 | 2018-09-24 14:55:55 +0200 | [diff] [blame] | 126 | RsproPDU_t *rspro_gen_ConnectClientRes(const struct app_comp_id *a_cid, e_ResultCode res) |
| 127 | { |
| 128 | RsproPDU_t *pdu = CALLOC(1, sizeof(*pdu)); |
| 129 | if (!pdu) |
| 130 | return NULL; |
| 131 | pdu->version = 2; |
| 132 | pdu->tag = 2342; |
| 133 | pdu->msg.present = RsproPDUchoice_PR_connectClientRes; |
| 134 | fill_comp_id(&pdu->msg.choice.connectClientRes.identity, a_cid); |
| 135 | pdu->msg.choice.connectClientRes.result = res; |
| 136 | |
| 137 | return pdu; |
| 138 | } |
| 139 | |
Harald Welte | 3aa901d | 2018-08-13 18:32:36 +0200 | [diff] [blame] | 140 | RsproPDU_t *rspro_gen_CreateMappingReq(const ClientSlot_t *client, const BankSlot_t *bank) |
| 141 | { |
| 142 | RsproPDU_t *pdu = CALLOC(1, sizeof(*pdu)); |
| 143 | if (!pdu) |
| 144 | return NULL; |
| 145 | pdu->msg.present = RsproPDUchoice_PR_createMappingReq; |
| 146 | pdu->msg.choice.createMappingReq.client = *client; |
| 147 | pdu->msg.choice.createMappingReq.bank = *bank; |
| 148 | |
| 149 | return pdu; |
| 150 | } |
| 151 | |
Harald Welte | 371d026 | 2018-08-16 15:23:58 +0200 | [diff] [blame] | 152 | RsproPDU_t *rspro_gen_ConfigClientReq(const ClientSlot_t *client, uint32_t ip, uint16_t port) |
Harald Welte | 3aa901d | 2018-08-13 18:32:36 +0200 | [diff] [blame] | 153 | { |
| 154 | RsproPDU_t *pdu = CALLOC(1, sizeof(*pdu)); |
| 155 | if (!pdu) |
| 156 | return NULL; |
| 157 | pdu->msg.present = RsproPDUchoice_PR_configClientReq; |
Harald Welte | 371d026 | 2018-08-16 15:23:58 +0200 | [diff] [blame] | 158 | pdu->msg.choice.configClientReq.clientSlot = *client; |
Harald Welte | 3aa901d | 2018-08-13 18:32:36 +0200 | [diff] [blame] | 159 | fill_ip4_port(&pdu->msg.choice.configClientReq.bankd, ip, port); |
| 160 | |
| 161 | return pdu; |
| 162 | } |
| 163 | |
| 164 | RsproPDU_t *rspro_gen_SetAtrReq(uint16_t client_id, uint16_t slot_nr, const uint8_t *atr, |
| 165 | unsigned int atr_len) |
| 166 | { |
| 167 | RsproPDU_t *pdu = CALLOC(1, sizeof(*pdu)); |
| 168 | if (!pdu) |
| 169 | return NULL; |
| 170 | pdu->msg.present = RsproPDUchoice_PR_setAtrReq; |
| 171 | pdu->msg.choice.setAtrReq.slot.clientId = client_id; |
| 172 | pdu->msg.choice.setAtrReq.slot.slotNr = slot_nr; |
| 173 | OCTET_STRING_fromBuf(&pdu->msg.choice.setAtrReq.atr, (const char *)atr, atr_len); |
| 174 | |
| 175 | return pdu; |
| 176 | } |
Harald Welte | 5d16b1c | 2018-09-23 19:25:46 +0200 | [diff] [blame] | 177 | |
| 178 | RsproPDU_t *rspro_gen_TpduModem2Card(const ClientSlot_t *client, const BankSlot_t *bank, |
| 179 | const uint8_t *tpdu, unsigned int tpdu_len) |
| 180 | { |
| 181 | RsproPDU_t *pdu = CALLOC(1, sizeof(*pdu)); |
| 182 | if (!pdu) |
| 183 | return NULL; |
| 184 | pdu->msg.present = RsproPDUchoice_PR_tpduModemToCard; |
Kévin Redon | 9744bd9 | 2018-10-11 08:37:38 +0200 | [diff] [blame] | 185 | OSMO_ASSERT(client); |
Harald Welte | 5d16b1c | 2018-09-23 19:25:46 +0200 | [diff] [blame] | 186 | pdu->msg.choice.tpduModemToCard.fromClientSlot = *client; |
Kévin Redon | 9744bd9 | 2018-10-11 08:37:38 +0200 | [diff] [blame] | 187 | OSMO_ASSERT(bank); |
Harald Welte | 5d16b1c | 2018-09-23 19:25:46 +0200 | [diff] [blame] | 188 | pdu->msg.choice.tpduModemToCard.toBankSlot = *bank; |
| 189 | /* TODO: flags? */ |
| 190 | OCTET_STRING_fromBuf(&pdu->msg.choice.tpduModemToCard.data, (const char *)tpdu, tpdu_len); |
| 191 | |
| 192 | return pdu; |
| 193 | } |
| 194 | |
| 195 | RsproPDU_t *rspro_gen_TpduCard2Modem(const BankSlot_t *bank, const ClientSlot_t *client, |
| 196 | const uint8_t *tpdu, unsigned int tpdu_len) |
| 197 | { |
| 198 | RsproPDU_t *pdu = CALLOC(1, sizeof(*pdu)); |
| 199 | if (!pdu) |
| 200 | return NULL; |
| 201 | pdu->msg.present = RsproPDUchoice_PR_tpduCardToModem; |
Kévin Redon | 9744bd9 | 2018-10-11 08:37:38 +0200 | [diff] [blame] | 202 | OSMO_ASSERT(bank); |
Harald Welte | 5d16b1c | 2018-09-23 19:25:46 +0200 | [diff] [blame] | 203 | pdu->msg.choice.tpduCardToModem.fromBankSlot = *bank; |
Kévin Redon | 9744bd9 | 2018-10-11 08:37:38 +0200 | [diff] [blame] | 204 | OSMO_ASSERT(client) |
Harald Welte | 5d16b1c | 2018-09-23 19:25:46 +0200 | [diff] [blame] | 205 | pdu->msg.choice.tpduCardToModem.toClientSlot = *client; |
| 206 | /* TODO: flags? */ |
| 207 | OCTET_STRING_fromBuf(&pdu->msg.choice.tpduCardToModem.data, (const char *)tpdu, tpdu_len); |
| 208 | |
| 209 | return pdu; |
| 210 | } |