blob: a5dc6c954c1eccbe858bb4196fab95b8bfe7e011 [file] [log] [blame]
Harald Welte879ee412020-02-14 14:21:33 +01001/* (C) 2018-2020 by Harald Welte <laforge@gnumonks.org>
Harald Welte3dcdd202019-03-09 13:06:46 +01002 *
3 * All Rights Reserved
4 *
5 * SPDX-License-Identifier: GPL-2.0+
6 *
7 * This program is free software; you can redistribute it and/or modify
8 * it under the terms of the GNU General Public License as published by
9 * the Free Software Foundation; either version 2 of the License, or
10 * (at your option) any later version.
11 *
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU General Public License for more details.
16 *
Harald Welte3dcdd202019-03-09 13:06:46 +010017 */
Harald Welte2ff0ab92018-08-17 22:10:49 +020018
Harald Welte0e968cc2020-02-22 18:16:16 +010019/* This file contains code shared among all remsim client applications,
20 * including the ifd-handler, which is not an executable program with a main()
21 * function or command line parsing, but a shared library */
22
Harald Welte2ff0ab92018-08-17 22:10:49 +020023#include <errno.h>
24#include <string.h>
Harald Welte228af8a2019-03-08 22:20:21 +010025
Harald Welte2ff0ab92018-08-17 22:10:49 +020026#include <talloc.h>
27
Harald Welte2ff0ab92018-08-17 22:10:49 +020028#include <osmocom/core/fsm.h>
29#include <osmocom/core/utils.h>
30#include <osmocom/core/logging.h>
Harald Welte2ff0ab92018-08-17 22:10:49 +020031
32#include "rspro_util.h"
Harald Welte24173fb2018-08-24 20:37:28 +020033#include "client.h"
Harald Welte61d98e92019-03-03 15:43:07 +010034#include "debug.h"
Harald Welte2ff0ab92018-08-17 22:10:49 +020035
Harald Welte0e968cc2020-02-22 18:16:16 +010036struct client_config *client_config_init(void *ctx)
37{
38 struct client_config *cfg = talloc_zero(ctx, struct client_config);
39 if (!cfg)
40 return NULL;
41
42 cfg->server_host = talloc_strdup(cfg, "127.0.0.1");
43 cfg->server_port = 9998;
44 cfg->client_id = -1;
45 cfg->client_slot = -1;
46 cfg->gsmtap_host = talloc_strdup(cfg, "127.0.0.1");
47 cfg->keep_running = false;
48
49 cfg->usb.vendor_id = -1;
50 cfg->usb.product_id = -1;
51 cfg->usb.config_id = -1;
52 cfg->usb.if_num = -1;
53 cfg->usb.altsetting = 0;
54 cfg->usb.addr = -1;
55 cfg->usb.path = NULL;
56
57 cfg->atr.data[0] = 0x3B;
58 cfg->atr.data[1] = 0x00; // the shortest simplest ATR possible
59 cfg->atr.len = 2;
Harald Weltec8c77e02022-07-24 12:54:24 +020060 cfg->atr_ignore_rspro = false;
Harald Welte0e968cc2020-02-22 18:16:16 +010061
62 return cfg;
63};
64
Harald Welted5a87292020-02-13 16:44:56 +010065static int bankd_handle_rx(struct rspro_server_conn *bankdc, const RsproPDU_t *pdu)
Harald Welte2ff0ab92018-08-17 22:10:49 +020066{
Harald Welte0e968cc2020-02-22 18:16:16 +010067 struct bankd_client *bc = bankdc2bankd_client(bankdc);
68
Harald Welte24173fb2018-08-24 20:37:28 +020069 switch (pdu->msg.present) {
70 case RsproPDUchoice_PR_connectClientRes:
Harald Welte5ae0f312022-04-25 15:53:04 +020071 if (pdu->msg.choice.connectClientRes.identity.type != ComponentType_remsimBankd) {
72 LOGPFSML(bankdc->fi, LOGL_ERROR, "Server connection to a ComponentType(%ld) != RemsimBankd? "
73 "Check your IP/Port configuration\n",
74 pdu->msg.choice.connectClientRes.identity.type);
75 osmo_fsm_inst_dispatch(bankdc->fi, SRVC_E_DISCONNECT, NULL);
76 return -1;
77 }
Harald Weltee56f2b92019-03-02 17:02:13 +010078 /* Store 'identity' of bankd to in peer_comp_id */
Harald Welted5a87292020-02-13 16:44:56 +010079 rspro_comp_id_retrieve(&bankdc->peer_comp_id, &pdu->msg.choice.connectClientRes.identity);
80 osmo_fsm_inst_dispatch(bankdc->fi, SRVC_E_CLIENT_CONN_RES, (void *) pdu);
Harald Welte24173fb2018-08-24 20:37:28 +020081 break;
Harald Welted5a87292020-02-13 16:44:56 +010082 case RsproPDUchoice_PR_tpduCardToModem:
Harald Welte0e968cc2020-02-22 18:16:16 +010083 return osmo_fsm_inst_dispatch(bc->main_fi, MF_E_BANKD_TPDU, (void *) pdu);
Harald Welted5a87292020-02-13 16:44:56 +010084 case RsproPDUchoice_PR_setAtrReq:
Harald Welte0e968cc2020-02-22 18:16:16 +010085 return osmo_fsm_inst_dispatch(bc->main_fi, MF_E_BANKD_ATR, (void *) pdu);
86 case RsproPDUchoice_PR_bankSlotStatusInd:
87 return osmo_fsm_inst_dispatch(bc->main_fi, MF_E_BANKD_SLOT_STATUS, (void *) pdu);
Harald Welte24173fb2018-08-24 20:37:28 +020088 default:
Harald Welted5a87292020-02-13 16:44:56 +010089 LOGPFSML(bankdc->fi, LOGL_ERROR, "Unknown/Unsupported RSPRO PDU %s\n",
90 rspro_msgt_name(pdu));
Harald Welte24173fb2018-08-24 20:37:28 +020091 return -1;
92 }
93
94 return 0;
Harald Welte2ff0ab92018-08-17 22:10:49 +020095}
96
Harald Weltee56f2b92019-03-02 17:02:13 +010097/* handle incoming messages from server */
98static int srvc_handle_rx(struct rspro_server_conn *srvc, const RsproPDU_t *pdu)
99{
Harald Welte8e46ab62020-02-14 12:55:43 +0100100 struct bankd_client *bc = srvc2bankd_client(srvc);
Harald Weltee56f2b92019-03-02 17:02:13 +0100101 RsproPDU_t *resp;
102
103 switch (pdu->msg.present) {
104 case RsproPDUchoice_PR_connectClientRes:
Harald Welte5ae0f312022-04-25 15:53:04 +0200105 if (pdu->msg.choice.connectClientRes.identity.type != ComponentType_remsimServer) {
106 LOGPFSML(srvc->fi, LOGL_ERROR, "Server connection to a ComponentType(%ld) != RemsimServer? "
107 "Check your IP/Port configuration\n",
108 pdu->msg.choice.connectClientRes.identity.type);
109 osmo_fsm_inst_dispatch(srvc->fi, SRVC_E_DISCONNECT, NULL);
110 return -1;
111 }
Harald Weltee56f2b92019-03-02 17:02:13 +0100112 /* Store 'identity' of server in srvc->peer_comp_id */
113 rspro_comp_id_retrieve(&srvc->peer_comp_id, &pdu->msg.choice.connectClientRes.identity);
114 osmo_fsm_inst_dispatch(srvc->fi, SRVC_E_CLIENT_CONN_RES, (void *) pdu);
115 break;
Harald Welted571a3e2019-03-11 22:09:50 +0100116 case RsproPDUchoice_PR_configClientIdReq:
Harald Weltee56f2b92019-03-02 17:02:13 +0100117 /* store/set the clientID as instructed by the server */
Harald Welte8e46ab62020-02-14 12:55:43 +0100118 if (!srvc->clslot)
119 srvc->clslot = talloc_zero(srvc, ClientSlot_t);
120 *srvc->clslot = pdu->msg.choice.configClientIdReq.clientSlot;
121 if (!bc->bankd_conn.clslot)
122 bc->bankd_conn.clslot = talloc_zero(bc, ClientSlot_t);
123 *bc->bankd_conn.clslot = *bc->srv_conn.clslot;
Harald Welted571a3e2019-03-11 22:09:50 +0100124 /* send response to server */
125 resp = rspro_gen_ConfigClientIdRes(ResultCode_ok);
126 server_conn_send_rspro(srvc, resp);
127 break;
128 case RsproPDUchoice_PR_configClientBankReq:
Harald Welte0e968cc2020-02-22 18:16:16 +0100129 osmo_fsm_inst_dispatch(bc->main_fi, MF_E_SRVC_CONFIG_BANK, (void *) pdu);
Harald Weltee56f2b92019-03-02 17:02:13 +0100130 break;
131 default:
Harald Welte8d8d4f12019-03-27 22:50:39 +0100132 LOGPFSML(srvc->fi, LOGL_ERROR, "Unknown/Unsupported RSPRO PDU type: %s\n",
133 rspro_msgt_name(pdu));
Harald Weltee56f2b92019-03-02 17:02:13 +0100134 return -1;
135 }
136
137 return 0;
138}
139
Harald Welte0e968cc2020-02-22 18:16:16 +0100140struct bankd_client *remsim_client_create(void *ctx, const char *name, const char *software,
141 struct client_config *cfg)
Harald Welte879ee412020-02-14 14:21:33 +0100142{
143 struct bankd_client *bc = talloc_zero(ctx, struct bankd_client);
144 struct rspro_server_conn *srvc, *bankdc;
145 int rc;
146
147 if (!bc)
148 return NULL;
149
Harald Welte0e968cc2020-02-22 18:16:16 +0100150 bc->cfg = cfg;
151
152 bc->main_fi = main_fsm_alloc(bc, bc);
153 if (!bc->main_fi) {
Harald Welte7293e7b2021-12-08 21:29:11 +0100154 LOGP(DMAIN, LOGL_FATAL, "Unable to create main client FSM: %s\n", strerror(errno));
Harald Welte0e968cc2020-02-22 18:16:16 +0100155 exit(1);
156 }
157
158 remsim_client_set_clslot(bc, cfg->client_id, cfg->client_slot);
159
Harald Welte879ee412020-02-14 14:21:33 +0100160 /* create and [attempt to] establish connection to remsim-server */
161 srvc = &bc->srv_conn;
Harald Welte0e968cc2020-02-22 18:16:16 +0100162 srvc->server_host = cfg->server_host;
163 srvc->server_port = cfg->server_port;
Harald Welte879ee412020-02-14 14:21:33 +0100164 srvc->handle_rx = srvc_handle_rx;
165 srvc->own_comp_id.type = ComponentType_remsimClient;
166 OSMO_STRLCPY_ARRAY(srvc->own_comp_id.name, name);
167 OSMO_STRLCPY_ARRAY(srvc->own_comp_id.software, software);
168 OSMO_STRLCPY_ARRAY(srvc->own_comp_id.sw_version, PACKAGE_VERSION);
169
170 rc = server_conn_fsm_alloc(bc, srvc);
171 if (rc < 0) {
Harald Welte7293e7b2021-12-08 21:29:11 +0100172 LOGP(DMAIN, LOGL_FATAL, "Unable to create Server conn FSM: %s\n", strerror(errno));
Harald Welte879ee412020-02-14 14:21:33 +0100173 exit(1);
174 }
Harald Welte0e968cc2020-02-22 18:16:16 +0100175 osmo_fsm_inst_change_parent(srvc->fi, bc->main_fi, MF_E_SRVC_LOST);
176 srvc->parent_conn_evt = MF_E_SRVC_CONNECTED;
177 srvc->parent_disc_evt = MF_E_SRVC_LOST;
Harald Welte879ee412020-02-14 14:21:33 +0100178
179 bankdc = &bc->bankd_conn;
180 /* server_host / server_port are configured from remsim-server */
181 bankdc->handle_rx = bankd_handle_rx;
182 memcpy(&bankdc->own_comp_id, &srvc->own_comp_id, sizeof(bankdc->own_comp_id));
183 rc = server_conn_fsm_alloc(bc, bankdc);
184 if (rc < 0) {
Harald Welte7293e7b2021-12-08 21:29:11 +0100185 LOGP(DMAIN, LOGL_FATAL, "Unable to connect bankd conn FSM: %s\n", strerror(errno));
Harald Welte879ee412020-02-14 14:21:33 +0100186 exit(1);
187 }
188 osmo_fsm_inst_update_id(bankdc->fi, "bankd");
Harald Welte0e968cc2020-02-22 18:16:16 +0100189 osmo_fsm_inst_change_parent(bankdc->fi, bc->main_fi, MF_E_BANKD_LOST);
190 bankdc->parent_conn_evt = MF_E_BANKD_CONNECTED;
191 bankdc->parent_disc_evt = MF_E_BANKD_LOST;
Harald Welte879ee412020-02-14 14:21:33 +0100192
193 return bc;
194}
195
196void remsim_client_set_clslot(struct bankd_client *bc, int client_id, int slot_nr)
197{
198 if (!bc->srv_conn.clslot) {
199 bc->srv_conn.clslot = talloc_zero(bc, ClientSlot_t);
200 OSMO_ASSERT(bc->srv_conn.clslot);
201 }
202
203 if (!bc->bankd_conn.clslot) {
204 bc->bankd_conn.clslot = talloc_zero(bc, ClientSlot_t);
205 OSMO_ASSERT(bc->bankd_conn.clslot);
206 }
207
208 if (client_id >= 0) {
209 bc->srv_conn.clslot->clientId = client_id;
210 bc->bankd_conn.clslot->clientId = client_id;
211 }
212
213 if (slot_nr >= 0) {
214 bc->srv_conn.clslot->slotNr = slot_nr;
215 bc->bankd_conn.clslot->slotNr = slot_nr;
216 }
217}