blob: 247ae42a394edb65128b4bf95162a65065e82561 [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;
60
61 return cfg;
62};
63
Harald Welted5a87292020-02-13 16:44:56 +010064static int bankd_handle_rx(struct rspro_server_conn *bankdc, const RsproPDU_t *pdu)
Harald Welte2ff0ab92018-08-17 22:10:49 +020065{
Harald Welte0e968cc2020-02-22 18:16:16 +010066 struct bankd_client *bc = bankdc2bankd_client(bankdc);
67
Harald Welte24173fb2018-08-24 20:37:28 +020068 switch (pdu->msg.present) {
69 case RsproPDUchoice_PR_connectClientRes:
Harald Welte5ae0f312022-04-25 15:53:04 +020070 if (pdu->msg.choice.connectClientRes.identity.type != ComponentType_remsimBankd) {
71 LOGPFSML(bankdc->fi, LOGL_ERROR, "Server connection to a ComponentType(%ld) != RemsimBankd? "
72 "Check your IP/Port configuration\n",
73 pdu->msg.choice.connectClientRes.identity.type);
74 osmo_fsm_inst_dispatch(bankdc->fi, SRVC_E_DISCONNECT, NULL);
75 return -1;
76 }
Harald Weltee56f2b92019-03-02 17:02:13 +010077 /* Store 'identity' of bankd to in peer_comp_id */
Harald Welted5a87292020-02-13 16:44:56 +010078 rspro_comp_id_retrieve(&bankdc->peer_comp_id, &pdu->msg.choice.connectClientRes.identity);
79 osmo_fsm_inst_dispatch(bankdc->fi, SRVC_E_CLIENT_CONN_RES, (void *) pdu);
Harald Welte24173fb2018-08-24 20:37:28 +020080 break;
Harald Welted5a87292020-02-13 16:44:56 +010081 case RsproPDUchoice_PR_tpduCardToModem:
Harald Welte0e968cc2020-02-22 18:16:16 +010082 return osmo_fsm_inst_dispatch(bc->main_fi, MF_E_BANKD_TPDU, (void *) pdu);
Harald Welted5a87292020-02-13 16:44:56 +010083 case RsproPDUchoice_PR_setAtrReq:
Harald Welte0e968cc2020-02-22 18:16:16 +010084 return osmo_fsm_inst_dispatch(bc->main_fi, MF_E_BANKD_ATR, (void *) pdu);
85 case RsproPDUchoice_PR_bankSlotStatusInd:
86 return osmo_fsm_inst_dispatch(bc->main_fi, MF_E_BANKD_SLOT_STATUS, (void *) pdu);
Harald Welte24173fb2018-08-24 20:37:28 +020087 default:
Harald Welted5a87292020-02-13 16:44:56 +010088 LOGPFSML(bankdc->fi, LOGL_ERROR, "Unknown/Unsupported RSPRO PDU %s\n",
89 rspro_msgt_name(pdu));
Harald Welte24173fb2018-08-24 20:37:28 +020090 return -1;
91 }
92
93 return 0;
Harald Welte2ff0ab92018-08-17 22:10:49 +020094}
95
Harald Weltee56f2b92019-03-02 17:02:13 +010096/* handle incoming messages from server */
97static int srvc_handle_rx(struct rspro_server_conn *srvc, const RsproPDU_t *pdu)
98{
Harald Welte8e46ab62020-02-14 12:55:43 +010099 struct bankd_client *bc = srvc2bankd_client(srvc);
Harald Weltee56f2b92019-03-02 17:02:13 +0100100 RsproPDU_t *resp;
101
102 switch (pdu->msg.present) {
103 case RsproPDUchoice_PR_connectClientRes:
Harald Welte5ae0f312022-04-25 15:53:04 +0200104 if (pdu->msg.choice.connectClientRes.identity.type != ComponentType_remsimServer) {
105 LOGPFSML(srvc->fi, LOGL_ERROR, "Server connection to a ComponentType(%ld) != RemsimServer? "
106 "Check your IP/Port configuration\n",
107 pdu->msg.choice.connectClientRes.identity.type);
108 osmo_fsm_inst_dispatch(srvc->fi, SRVC_E_DISCONNECT, NULL);
109 return -1;
110 }
Harald Weltee56f2b92019-03-02 17:02:13 +0100111 /* Store 'identity' of server in srvc->peer_comp_id */
112 rspro_comp_id_retrieve(&srvc->peer_comp_id, &pdu->msg.choice.connectClientRes.identity);
113 osmo_fsm_inst_dispatch(srvc->fi, SRVC_E_CLIENT_CONN_RES, (void *) pdu);
114 break;
Harald Welted571a3e2019-03-11 22:09:50 +0100115 case RsproPDUchoice_PR_configClientIdReq:
Harald Weltee56f2b92019-03-02 17:02:13 +0100116 /* store/set the clientID as instructed by the server */
Harald Welte8e46ab62020-02-14 12:55:43 +0100117 if (!srvc->clslot)
118 srvc->clslot = talloc_zero(srvc, ClientSlot_t);
119 *srvc->clslot = pdu->msg.choice.configClientIdReq.clientSlot;
120 if (!bc->bankd_conn.clslot)
121 bc->bankd_conn.clslot = talloc_zero(bc, ClientSlot_t);
122 *bc->bankd_conn.clslot = *bc->srv_conn.clslot;
Harald Welted571a3e2019-03-11 22:09:50 +0100123 /* send response to server */
124 resp = rspro_gen_ConfigClientIdRes(ResultCode_ok);
125 server_conn_send_rspro(srvc, resp);
126 break;
127 case RsproPDUchoice_PR_configClientBankReq:
Harald Welte0e968cc2020-02-22 18:16:16 +0100128 osmo_fsm_inst_dispatch(bc->main_fi, MF_E_SRVC_CONFIG_BANK, (void *) pdu);
Harald Weltee56f2b92019-03-02 17:02:13 +0100129 break;
130 default:
Harald Welte8d8d4f12019-03-27 22:50:39 +0100131 LOGPFSML(srvc->fi, LOGL_ERROR, "Unknown/Unsupported RSPRO PDU type: %s\n",
132 rspro_msgt_name(pdu));
Harald Weltee56f2b92019-03-02 17:02:13 +0100133 return -1;
134 }
135
136 return 0;
137}
138
Harald Welte0e968cc2020-02-22 18:16:16 +0100139struct bankd_client *remsim_client_create(void *ctx, const char *name, const char *software,
140 struct client_config *cfg)
Harald Welte879ee412020-02-14 14:21:33 +0100141{
142 struct bankd_client *bc = talloc_zero(ctx, struct bankd_client);
143 struct rspro_server_conn *srvc, *bankdc;
144 int rc;
145
146 if (!bc)
147 return NULL;
148
Harald Welte0e968cc2020-02-22 18:16:16 +0100149 bc->cfg = cfg;
150
151 bc->main_fi = main_fsm_alloc(bc, bc);
152 if (!bc->main_fi) {
Harald Welte7293e7b2021-12-08 21:29:11 +0100153 LOGP(DMAIN, LOGL_FATAL, "Unable to create main client FSM: %s\n", strerror(errno));
Harald Welte0e968cc2020-02-22 18:16:16 +0100154 exit(1);
155 }
156
157 remsim_client_set_clslot(bc, cfg->client_id, cfg->client_slot);
158
Harald Welte879ee412020-02-14 14:21:33 +0100159 /* create and [attempt to] establish connection to remsim-server */
160 srvc = &bc->srv_conn;
Harald Welte0e968cc2020-02-22 18:16:16 +0100161 srvc->server_host = cfg->server_host;
162 srvc->server_port = cfg->server_port;
Harald Welte879ee412020-02-14 14:21:33 +0100163 srvc->handle_rx = srvc_handle_rx;
164 srvc->own_comp_id.type = ComponentType_remsimClient;
165 OSMO_STRLCPY_ARRAY(srvc->own_comp_id.name, name);
166 OSMO_STRLCPY_ARRAY(srvc->own_comp_id.software, software);
167 OSMO_STRLCPY_ARRAY(srvc->own_comp_id.sw_version, PACKAGE_VERSION);
168
169 rc = server_conn_fsm_alloc(bc, srvc);
170 if (rc < 0) {
Harald Welte7293e7b2021-12-08 21:29:11 +0100171 LOGP(DMAIN, LOGL_FATAL, "Unable to create Server conn FSM: %s\n", strerror(errno));
Harald Welte879ee412020-02-14 14:21:33 +0100172 exit(1);
173 }
Harald Welte0e968cc2020-02-22 18:16:16 +0100174 osmo_fsm_inst_change_parent(srvc->fi, bc->main_fi, MF_E_SRVC_LOST);
175 srvc->parent_conn_evt = MF_E_SRVC_CONNECTED;
176 srvc->parent_disc_evt = MF_E_SRVC_LOST;
Harald Welte879ee412020-02-14 14:21:33 +0100177
178 bankdc = &bc->bankd_conn;
179 /* server_host / server_port are configured from remsim-server */
180 bankdc->handle_rx = bankd_handle_rx;
181 memcpy(&bankdc->own_comp_id, &srvc->own_comp_id, sizeof(bankdc->own_comp_id));
182 rc = server_conn_fsm_alloc(bc, bankdc);
183 if (rc < 0) {
Harald Welte7293e7b2021-12-08 21:29:11 +0100184 LOGP(DMAIN, LOGL_FATAL, "Unable to connect bankd conn FSM: %s\n", strerror(errno));
Harald Welte879ee412020-02-14 14:21:33 +0100185 exit(1);
186 }
187 osmo_fsm_inst_update_id(bankdc->fi, "bankd");
Harald Welte0e968cc2020-02-22 18:16:16 +0100188 osmo_fsm_inst_change_parent(bankdc->fi, bc->main_fi, MF_E_BANKD_LOST);
189 bankdc->parent_conn_evt = MF_E_BANKD_CONNECTED;
190 bankdc->parent_disc_evt = MF_E_BANKD_LOST;
Harald Welte879ee412020-02-14 14:21:33 +0100191
192 return bc;
193}
194
195void remsim_client_set_clslot(struct bankd_client *bc, int client_id, int slot_nr)
196{
197 if (!bc->srv_conn.clslot) {
198 bc->srv_conn.clslot = talloc_zero(bc, ClientSlot_t);
199 OSMO_ASSERT(bc->srv_conn.clslot);
200 }
201
202 if (!bc->bankd_conn.clslot) {
203 bc->bankd_conn.clslot = talloc_zero(bc, ClientSlot_t);
204 OSMO_ASSERT(bc->bankd_conn.clslot);
205 }
206
207 if (client_id >= 0) {
208 bc->srv_conn.clslot->clientId = client_id;
209 bc->bankd_conn.clslot->clientId = client_id;
210 }
211
212 if (slot_nr >= 0) {
213 bc->srv_conn.clslot->slotNr = slot_nr;
214 bc->bankd_conn.clslot->slotNr = slot_nr;
215 }
216}