blob: 5bc7aeacfb70e3a517d49b195a041f15ebd8dd2f [file] [log] [blame]
Nikias Bassenf8066bb2013-02-26 17:24:23 +01001/*
2 * service.c
3 * generic service implementation.
4 *
5 * Copyright (c) 2013 Nikias Bassen. All Rights Reserved.
6 *
7 * This library is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU Lesser General Public
9 * License as published by the Free Software Foundation; either
10 * version 2.1 of the License, or (at your option) any later version.
11 *
12 * This library 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 GNU
15 * Lesser General Public License for more details.
16 *
17 * You should have received a copy of the GNU Lesser General Public
18 * License along with this library; if not, write to the Free Software
19 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
20 */
21#ifdef HAVE_CONFIG_H
22#include <config.h>
23#endif
24#include <stdlib.h>
25#include <string.h>
26
27#include "service.h"
28#include "idevice.h"
Martin Szulecki4748e1a2013-04-25 17:43:54 +010029#include "common/debug.h"
Nikias Bassenf8066bb2013-02-26 17:24:23 +010030
31/**
32 * Convert an idevice_error_t value to an service_error_t value.
33 * Used internally to get correct error codes.
34 *
35 * @param err An idevice_error_t error code
36 *
37 * @return A matching service_error_t error code,
38 * SERVICE_E_UNKNOWN_ERROR otherwise.
39 */
40static service_error_t idevice_to_service_error(idevice_error_t err)
41{
42 switch (err) {
43 case IDEVICE_E_SUCCESS:
44 return SERVICE_E_SUCCESS;
45 case IDEVICE_E_INVALID_ARG:
46 return SERVICE_E_INVALID_ARG;
47 case IDEVICE_E_SSL_ERROR:
48 return SERVICE_E_SSL_ERROR;
49 default:
50 break;
51 }
52 return SERVICE_E_UNKNOWN_ERROR;
53}
54
55/**
56 * Creates a new service for the specified service descriptor.
57 *
58 * @param device The device to connect to.
59 * @param service The service descriptor returned by lockdownd_start_service.
60 * @param client Pointer that will be set to a newly allocated
61 * service_client_t upon successful return.
62 *
63 * @return SERVICE_E_SUCCESS on success,
64 * SERVICE_E_INVALID_ARG when one of the arguments is invalid,
65 * or SERVICE_E_MUX_ERROR when connecting to the device failed.
66 */
67service_error_t service_client_new(idevice_t device, lockdownd_service_descriptor_t service, service_client_t *client)
68{
Nikias Bassen4222bc12013-03-14 03:09:14 +010069 if (!device || !service || service->port == 0 || !client || *client)
Nikias Bassenf8066bb2013-02-26 17:24:23 +010070 return SERVICE_E_INVALID_ARG;
71
72 /* Attempt connection */
73 idevice_connection_t connection = NULL;
74 if (idevice_connect(device, service->port, &connection) != IDEVICE_E_SUCCESS) {
75 return SERVICE_E_MUX_ERROR;
76 }
77
78 /* create client object */
79 service_client_t client_loc = (service_client_t)malloc(sizeof(struct service_client_private));
80 client_loc->connection = connection;
81
82 /* enable SSL if requested */
83 if (service->ssl_enabled == 1)
84 service_enable_ssl(client_loc);
85
86 /* all done, return success */
87 *client = client_loc;
88 return SERVICE_E_SUCCESS;
89}
90
91/**
92 * Starts a new service on the specified device with given name and
93 * connects to it.
94 *
95 * @param device The device to connect to.
96 * @param service_name The name of the service to start.
97 * @param client Pointer that will point to a newly allocated service_client_t
98 * upon successful return. Must be freed using service_client_free() after
99 * use.
Martin Szuleckia3cdb722013-02-26 20:07:53 +0100100 * @param label The label to use for communication. Usually the program name.
101 * Pass NULL to disable sending the label in requests to lockdownd.
Nikias Bassenf8066bb2013-02-26 17:24:23 +0100102 *
103 * @return SERVICE_E_SUCCESS on success, or a SERVICE_E_* error code
104 * otherwise.
105 */
Nikias Bassen31f24dd2013-02-28 02:17:27 +0100106service_error_t service_client_factory_start_service(idevice_t device, const char* service_name, void **client, const char* label, int16_t (*constructor_func)(idevice_t, lockdownd_service_descriptor_t, void**), int16_t *error_code)
Nikias Bassenf8066bb2013-02-26 17:24:23 +0100107{
108 *client = NULL;
109
110 lockdownd_client_t lckd = NULL;
Martin Szuleckia3cdb722013-02-26 20:07:53 +0100111 if (LOCKDOWN_E_SUCCESS != lockdownd_client_new_with_handshake(device, &lckd, label)) {
Nikias Bassenf8066bb2013-02-26 17:24:23 +0100112 debug_info("Could not create a lockdown client.");
113 return SERVICE_E_START_SERVICE_ERROR;
114 }
115
116 lockdownd_service_descriptor_t service = NULL;
117 lockdownd_start_service(lckd, service_name, &service);
118 lockdownd_client_free(lckd);
119
Nikias Bassen4222bc12013-03-14 03:09:14 +0100120 if (!service || service->port == 0) {
Nikias Bassenf8066bb2013-02-26 17:24:23 +0100121 debug_info("Could not start service %s!", service_name);
122 return SERVICE_E_START_SERVICE_ERROR;
123 }
124
Nikias Bassen31f24dd2013-02-28 02:17:27 +0100125 int16_t ec;
Nikias Bassendbe40172013-02-27 15:50:42 +0100126 if (constructor_func) {
Nikias Bassen31f24dd2013-02-28 02:17:27 +0100127 ec = (int16_t)constructor_func(device, service, client);
Nikias Bassendbe40172013-02-27 15:50:42 +0100128 } else {
129 ec = service_client_new(device, service, (service_client_t*)client);
130 }
131 if (error_code) {
132 *error_code = ec;
Nikias Bassenf8066bb2013-02-26 17:24:23 +0100133 }
134
Nikias Bassendbe40172013-02-27 15:50:42 +0100135 if (ec != SERVICE_E_SUCCESS) {
136 debug_info("Could not connect to service %s! Port: %i, error: %i", service_name, service->port, ec);
Nikias Bassenf8066bb2013-02-26 17:24:23 +0100137 }
138
Nikias Bassendbe40172013-02-27 15:50:42 +0100139 lockdownd_service_descriptor_free(service);
140 service = NULL;
141
142 return (ec == SERVICE_E_SUCCESS) ? SERVICE_E_SUCCESS : SERVICE_E_START_SERVICE_ERROR;
Nikias Bassenf8066bb2013-02-26 17:24:23 +0100143}
144
145/**
146 * Frees a service instance.
147 *
148 * @param client The service instance to free.
149 *
150 * @return SERVICE_E_SUCCESS on success,
151 * SERVICE_E_INVALID_ARG when client is invalid, or a
152 * SERVICE_E_UNKNOWN_ERROR when another error occured.
153 */
154service_error_t service_client_free(service_client_t client)
155{
156 if (!client)
157 return SERVICE_E_INVALID_ARG;
158
159 service_error_t err = idevice_to_service_error(idevice_disconnect(client->connection));
160 free(client);
161 return err;
162}
163
164/**
165 * Sends data using the given service client.
166 *
167 * @param client The service client to use for sending.
168 * @param data Data to send
169 * @param size Size of the data to send
170 * @param sent Number of bytes sent (can be NULL to ignore)
171 *
172 * @return SERVICE_E_SUCCESS on success,
173 * SERVICE_E_INVALID_ARG when one or more parameters are
174 * invalid, or SERVICE_E_UNKNOWN_ERROR when an unspecified
175 * error occurs.
176 */
177service_error_t service_send(service_client_t client, const char* data, uint32_t size, uint32_t *sent)
178{
179 service_error_t res = SERVICE_E_UNKNOWN_ERROR;
180 int bytes = 0;
181
182 if (!client || (client && !client->connection) || !data || (size == 0)) {
183 return SERVICE_E_INVALID_ARG;
184 }
185
186 debug_info("sending %d bytes", size);
187 res = idevice_to_service_error(idevice_connection_send(client->connection, data, size, (uint32_t*)&bytes));
188 if (bytes <= 0) {
189 debug_info("ERROR: sending to device failed.");
190 }
191 if (sent) {
192 *sent = (uint32_t)bytes;
193 }
194
195 return res;
196}
197
198/**
199 * Receives data using the given service client with specified timeout.
200 *
201 * @param client The service client to use for receiving
202 * @param data Buffer that will be filled with the data received
203 * @param size Number of bytes to receive
204 * @param received Number of bytes received (can be NULL to ignore)
205 * @param timeout Maximum time in milliseconds to wait for data.
206 *
207 * @return SERVICE_E_SUCCESS on success,
208 * SERVICE_E_INVALID_ARG when one or more parameters are
209 * invalid, SERVICE_E_MUX_ERROR when a communication error
210 * occurs, or SERVICE_E_UNKNOWN_ERROR when an unspecified
211 * error occurs.
212 */
213service_error_t service_receive_with_timeout(service_client_t client, char* data, uint32_t size, uint32_t *received, unsigned int timeout)
214{
215 service_error_t res = SERVICE_E_UNKNOWN_ERROR;
216 int bytes = 0;
217
218 if (!client || (client && !client->connection) || !data || (size == 0)) {
219 return SERVICE_E_INVALID_ARG;
220 }
221
222 res = idevice_to_service_error(idevice_connection_receive_timeout(client->connection, data, size, (uint32_t*)&bytes, timeout));
223 if (bytes <= 0) {
224 debug_info("could not read data");
225 }
226 if (received) {
227 *received = (uint32_t)bytes;
228 }
229
230 return res;
231}
232
233/**
234 * Receives data using the given service client.
235 *
236 * @param client The service client to use for receiving
237 * @param data Buffer that will be filled with the data received
238 * @param size Number of bytes to receive
239 * @param received Number of bytes received (can be NULL to ignore)
240 *
241 * @return SERVICE_E_SUCCESS on success,
242 * SERVICE_E_INVALID_ARG when one or more parameters are
243 * invalid, SERVICE_E_MUX_ERROR when a communication error
244 * occurs, or SERVICE_E_UNKNOWN_ERROR when an unspecified
245 * error occurs.
246 */
247service_error_t service_receive(service_client_t client, char* data, uint32_t size, uint32_t *received)
248{
249 return service_receive_with_timeout(client, data, size, received, 10000);
250}
251
252/**
253 * Enable SSL for the given service client.
254 *
255 * @param client The connected service client for that SSL should be enabled.
256 *
257 * @return SERVICE_E_SUCCESS on success,
258 * SERVICE_E_INVALID_ARG if client or client->connection is
259 * NULL, SERVICE_E_SSL_ERROR when SSL could not be enabled,
260 * or SERVICE_E_UNKNOWN_ERROR otherwise.
261 */
262service_error_t service_enable_ssl(service_client_t client)
263{
264 if (!client || !client->connection)
265 return SERVICE_E_INVALID_ARG;
266 return idevice_to_service_error(idevice_connection_enable_ssl(client->connection));
267}
268
269/**
270 * Disable SSL for the given service client.
271 *
272 * @param client The connected service client for that SSL should be disabled.
273 *
274 * @return SERVICE_E_SUCCESS on success,
275 * SERVICE_E_INVALID_ARG if client or client->connection is
276 * NULL, or SERVICE_E_UNKNOWN_ERROR otherwise.
277 */
278service_error_t service_disable_ssl(service_client_t client)
279{
280 if (!client || !client->connection)
281 return SERVICE_E_INVALID_ARG;
282 return idevice_to_service_error(idevice_connection_disable_ssl(client->connection));
283}
284