1 //
2 // Copyright (C) 2020 The Android Open Source Project
3 //
4 // Licensed under the Apache License, Version 2.0 (the "License");
5 // you may not use this file except in compliance with the License.
6 // You may obtain a copy of the License at
7 //
8 // http://www.apache.org/licenses/LICENSE-2.0
9 //
10 // Unless required by applicable law or agreed to in writing, software
11 // distributed under the License is distributed on an "AS IS" BASIS,
12 // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 // See the License for the specific language governing permissions and
14 // limitations under the License.
15
16 #include "host/commands/modem_simulator/sup_service.h"
17
18 namespace cuttlefish {
19
SupService(int32_t service_id,ChannelMonitor * channel_monitor,ThreadLooper * thread_looper)20 SupService::SupService(int32_t service_id, ChannelMonitor* channel_monitor,
21 ThreadLooper* thread_looper)
22 : ModemService(service_id, this->InitializeCommandHandlers(),
23 channel_monitor, thread_looper) {
24 InitializeServiceState();
25 }
26
InitializeCommandHandlers()27 std::vector<CommandHandler> SupService::InitializeCommandHandlers() {
28 std::vector<CommandHandler> command_handlers = {
29 CommandHandler("+CUSD",
30 [this](const Client& client, std::string& cmd) {
31 this->HandleUSSD(client, cmd);
32 }),
33 CommandHandler("+CLIR",
34 [this](const Client& client, std::string& cmd) {
35 this->HandleCLIR(client, cmd);
36 }),
37 CommandHandler("+CCWA",
38 [this](const Client& client, std::string& cmd) {
39 this->HandleCallWaiting(client, cmd);
40 }),
41 CommandHandler(
42 "+CLIP?", [this](const Client& client) { this->HandleCLIP(client); }),
43 CommandHandler("+CCFCU",
44 [this](const Client& client, std::string& cmd) {
45 this->HandleCallForward(client, cmd);
46 }),
47 CommandHandler("+CSSN",
48 [this](const Client& client, std::string& cmd) {
49 this->HandleSuppServiceNotifications(client, cmd);
50 }),
51 };
52 return (command_handlers);
53 }
54
InitializeServiceState()55 void SupService::InitializeServiceState() {
56 call_forward_infos_ = {
57 CallForwardInfo(CallForwardInfo::Reason::CFU),
58 CallForwardInfo(CallForwardInfo::Reason::CFB),
59 CallForwardInfo(CallForwardInfo::Reason::CFNR),
60 CallForwardInfo(CallForwardInfo::Reason::CFNRC)
61 };
62 }
63
64 /**
65 * AT+CUSD
66 * This command allows control of the Unstructured Supplementary Service Data (USSD)
67 * according to 3GPP TS 22.090 [23], 3GPP TS 24.090 [148] and 3GPP TS 24.390 [131].
68 * Both network and mobile initiated operations are supported.
69 *
70 * Command Possible response(s)
71 * +CUSD=[<n>[,<str>[,<dcs>]]] +CME ERROR: <err>
72 * +CUSD? +CUSD: <n>
73 *
74 * <n>: integer type (sets/shows the result code presentation status to the TE).
75 * 0 disable the result code presentation to the TE
76 * 1 enable the result code presentation to the TE
77 * 2 cancel session (not applicable to read command response)
78 * <str>: string type USSD string
79 * when <str> parameter is not given, network is not interrogated
80 * <dcs>: integer type (shows Cell Broadcast Data Coding Scheme, see 3GPP TS 23.038 [25]).
81 * Default value is 0.
82 *
83 * see RIL_REQUEST_SEND_USSD or RIL_REQUEST_CANCEL_USSD in RIL
84 */
HandleUSSD(const Client & client,std::string &)85 void SupService::HandleUSSD(const Client& client, std::string& /*command*/) {
86 client.SendCommandResponse("OK");
87 }
88
89 /**
90 * AT+CLIR
91 * This command refers to CLIR‑service according to 3GPP TS 22.081 that allows
92 * a calling subscriber to enable or disable the presentation of the CLI to the
93 * called party when originating a call.
94 *
95 * Command Possible response(s)
96 * +CLIR: <n>
97 * +CLIR? +CLIR: <n>,<m>
98 *
99 * <n>: integer type (parameter sets the adjustment for outgoing calls).
100 * 0 presentation indicator is used according to the subscription of the CLIR service
101 * 1 CLIR invocation
102 * 2 CLIR suppression
103 * <m>: integer type (parameter shows the subscriber CLIR / OIR service status in the network).
104 * 0 CLIR / OIR not provisioned
105 * 1 CLIR / OIR provisioned in permanent mode
106 * 2 unknown (e.g. no network, etc.)
107 * 3 CLIR / OIR temporary mode presentation restricted
108 * 4 CLIR / OIR temporary mode presentation allowed
109 *
110 * see RIL_REQUEST_SET_CLIR or RIL_REQUEST_GET_CLIR in RIL
111 */
HandleCLIR(const Client & client,std::string & command)112 void SupService::HandleCLIR(const Client& client, std::string& command) {
113 std::vector<std::string> responses;
114 std::stringstream ss;
115
116 CommandParser cmd(command);
117 cmd.SkipPrefix();
118 if (*cmd == "AT+CLIR?") {
119 ss << "+CLIR:" << clir_status_.type << "," << clir_status_.status;
120 responses.push_back(ss.str());
121 } else {
122 clir_status_.type = (ClirStatusInfo::ClirType)cmd.GetNextInt();
123 }
124 responses.push_back("OK");
125 client.SendCommandResponse(responses);
126 }
127
128 /**
129 * AT+CLIP
130 * This command refers to the supplementary service CLIP (Calling Line
131 * Identification Presentation) according to 3GPP TS 22.081 [3] and OIP
132 * (Originating Identification Presentation) according to 3GPP TS 24.607 [119]
133 * that enables a called subscriber to get the calling line identity (CLI) of
134 * the calling party when receiving a mobile terminated call.
135 *
136 * Command Possible response(s)
137 * +CLIP? +CLIP: <n>,<m>
138 *
139 * <n>: integer type (parameter sets/shows the result code presentation status to the TE).
140 * 0 disable
141 * 1 enable
142 * <m>: integer type (parameter shows the subscriber CLIR / OIR service status in the network).
143 * 0 CLIP / OIP not provisioned
144 * 1 CLIP / OIP provisioned
145 * 2 unknown (e.g. no network, etc.)
146 *
147 * see RIL_REQUEST_QUERY_CLIP in RIL
148 */
HandleCLIP(const Client & client)149 void SupService::HandleCLIP(const Client& client) {
150 std::vector<std::string> responses = {"+CLIP: 0, 0", "OK"};
151 client.SendCommandResponse(responses);
152 }
153
154 /**
155 * AT+CSSN
156 * This command refers to supplementary service related network initiated
157 * notifications. The set command enables/disables the presentation of
158 * notification result codes from TA to TE.
159 *
160 * Command Possible response(s)
161 * +CSSN: [<n>[,<m>]]
162 *
163 * <n>: integer type (parameter sets/shows the +CSSI intermediate result code
164 * presentation status to the TE)
165 * 0 disable
166 * 1 enable
167 * <m>: integer type (parameter sets/shows the +CSSU unsolicited result code
168 * presentation status to the TE)
169 * 0 disable
170 * 1 enable
171 *
172 * see RIL_REQUEST_SET_SUPP_SVC_NOTIFICATION in RIL
173 */
HandleSuppServiceNotifications(const Client & client,std::string &)174 void SupService::HandleSuppServiceNotifications(const Client& client, std::string& /*command*/) {
175 client.SendCommandResponse("OK");
176 }
177
178 /**
179 * AT+CCFCU
180 * The command allows control of the communication forwarding supplementary service
181 * according to 3GPP TS 22.072 [31], 3GPP TS 22.082 [4] and 3GPP TS 24.604 [132].
182 *
183 * Command Possible response(s)
184 * +CCFCU=<reason>,<mode> +CME ERROR: <err>
185 * [,<numbertype>,<ton>,<number> when <mode>=2 and command successful:
186 * [,<class>,<ruleset> +CCFCU: <status>,<class1>[,<numbertype>,
187 * [,<subaddr>[,<satype>[,<time>]]]]] <ton>,<number>[,<subaddr>,<satype>[,<time>]]]
188 * [,<class>,<ruleset>
189 *
190 * see SupService::CallForwardInfo
191 *
192 * see RIL_REQUEST_SET_CALL_FORWARD or RIL_REQUEST_QUERY_CALL_FORWARD_STATUS in RIL
193 */
HandleCallForward(const Client & client,std::string & command)194 void SupService::HandleCallForward(const Client& client, std::string& command) {
195 std::vector<std::string> responses;
196 std::stringstream ss;
197
198 CommandParser cmd(command);
199 cmd.SkipPrefix();
200
201 int reason = cmd.GetNextInt();
202 int status = cmd.GetNextInt();
203 int number_type = cmd.GetNextInt();
204 int ton = cmd.GetNextInt();
205 std::string_view number = cmd.GetNextStr();
206 int classx = cmd.GetNextInt();
207
208 switch (reason) {
209 case CallForwardInfo::Reason::ALL_CF: {
210 if (status == CallForwardInfo::CallForwardInfoStatus::INTERROGATE) {
211 auto iter = call_forward_infos_.begin();
212 for (; iter != call_forward_infos_.end(); ++iter) {
213 ss.clear();
214 ss << "+CCFCU: " << iter->status << "," << classx << "," << number_type
215 << "," << ton << ",\"" << iter->number << "\"";
216 if (iter->reason == CallForwardInfo::Reason::CFNR) {
217 ss << ",,," << iter->timeSeconds;
218 }
219 responses.push_back(ss.str());
220 ss.str("");
221 }
222 }
223 break;
224 }
225 case CallForwardInfo::Reason::CFU:
226 case CallForwardInfo::Reason::CFB:
227 case CallForwardInfo::Reason::CFNR:
228 case CallForwardInfo::Reason::CFNRC: {
229 if (status == CallForwardInfo::CallForwardInfoStatus::INTERROGATE) {
230 ss << "+CCFCU: " << call_forward_infos_[reason].status
231 << "," << classx << "," << number_type << "," << ton << ",\""
232 << call_forward_infos_[reason].number << "\"";
233 if (reason == CallForwardInfo::Reason::CFNR) {
234 ss << ",,," << call_forward_infos_[reason].timeSeconds;
235 }
236 responses.push_back(ss.str());
237 } else {
238 if (status == CallForwardInfo::CallForwardInfoStatus::REGISTRATION) {
239 call_forward_infos_[reason].status
240 = CallForwardInfo::CallForwardInfoStatus::ENABLE;
241 } else {
242 call_forward_infos_[reason].status =
243 (CallForwardInfo::CallForwardInfoStatus)status;
244 }
245 call_forward_infos_[reason].number_type = number_type;
246 call_forward_infos_[reason].ton = ton;
247 call_forward_infos_[reason].number = number;
248 if (reason == CallForwardInfo::Reason::CFNR) {
249 cmd.SkipComma();
250 cmd.SkipComma();
251 cmd.SkipComma();
252 int timeSeconds = cmd.GetNextInt();
253 call_forward_infos_[reason].timeSeconds = timeSeconds >= 0 ? timeSeconds : 0;
254 }
255 }
256 break;
257 }
258 default:
259 client.SendCommandResponse(kCmeErrorInCorrectParameters);
260 return;
261 }
262
263 responses.push_back("OK");
264 client.SendCommandResponse(responses);
265 }
266
267 /**
268 * AT+CCWA
269 * This command allows control of the supplementary service Call Waiting
270 * according to 3GPP TS 22.083 [5] and Communication Waiting according to
271 * 3GPP TS 24.607 [137]. Activation, deactivation and status query are supported.
272 *
273 * Command Possible response(s)
274 * +CCWA=[<n>[,<mode>[,<class>]]] +CME ERROR: <err>
275 * when <mode>=2 and command successful
276 +CCWA: <status>,<class1>
277 [<CR><LF>+CCWA: <status>,<class2>
278 * <n>: integer type (sets/shows the result code presentation status to the TE).
279 * 0 disable
280 * 1 enable
281 * <mode>: integer type (when <mode> parameter is not given, network is not interrogated).
282 * 0 disable
283 * 1 enable
284 * 2 query status
285 * <classx>: a sum of integers each representing a class of information
286 * (default 7 - voice, data and fax).
287 * <status>: integer type
288 * 0 not active
289 * 1 active
290 *
291 * see RIL_REQUEST_QUERY_CALL_WAITING and RIL_REQUEST_SET_CALL_WAITING in RIL
292 */
HandleCallWaiting(const Client & client,std::string & command)293 void SupService::HandleCallWaiting(const Client& client, std::string& command) {
294 std::vector<std::string> responses;
295 std::stringstream ss;
296
297 CommandParser cmd(command);
298 cmd.SkipPrefix();
299 cmd.SkipComma();
300 int mode = cmd.GetNextInt();
301 int classx = cmd.GetNextInt();
302
303 if (mode == 2) { // Query
304 if (classx == -1) {
305 classx = 7;
306 }
307 ss << "+CCWA: " << call_waiting_info_.mode << "," << classx;
308 responses.push_back(ss.str());
309 } else if (mode == 0 || mode == 1) { // Enable or disable
310 call_waiting_info_.mode = mode;
311 if (classx != -1) {
312 call_waiting_info_.classx = classx;
313 }
314 }
315
316 responses.push_back("OK");
317 client.SendCommandResponse(responses);
318 }
319
320 } // namespace cuttlefish
321