2 * Copyright (C) 2007-2008 Tobias Brunner
3 * Hochschule fuer Technik Rapperswil
5 * This program is free software; you can redistribute it and/or modify it
6 * under the terms of the GNU General Public License as published by the
7 * Free Software Foundation; either version 2 of the License, or (at your
8 * option) any later version. See <http://www.fsf.org/copyleft/gpl.txt>.
10 * This program is distributed in the hope that it will be useful, but
11 * WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
12 * or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
16 #include "initiate_mediation_job.h"
18 #include <sa/ike_sa.h>
22 typedef struct private_initiate_mediation_job_t private_initiate_mediation_job_t
;
25 * Private data of an initiate_mediation_job_t Object
27 struct private_initiate_mediation_job_t
{
29 * public initiate_mediation_job_t interface
31 initiate_mediation_job_t
public;
34 * ID of the IKE_SA of the mediated connection.
36 ike_sa_id_t
*mediated_sa_id
;
39 * ID of the IKE_SA of the mediation connection.
41 ike_sa_id_t
*mediation_sa_id
;
45 * Implements job_t.destroy.
47 static void destroy(private_initiate_mediation_job_t
*this)
49 DESTROY_IF(this->mediation_sa_id
);
50 DESTROY_IF(this->mediated_sa_id
);
55 * Callback to handle initiation of mediation connection
57 static bool initiate_callback(private_initiate_mediation_job_t
*this,
58 debug_t group
, level_t level
, ike_sa_t
*ike_sa
,
59 char *format
, va_list args
)
61 if (ike_sa
&& !this->mediation_sa_id
)
63 this->mediation_sa_id
= ike_sa
->get_id(ike_sa
);
64 this->mediation_sa_id
= this->mediation_sa_id
->clone(this->mediation_sa_id
);
70 * Implementation of job_t.execute.
72 static void initiate(private_initiate_mediation_job_t
*this)
74 ike_sa_t
*mediated_sa
, *mediation_sa
;
75 peer_cfg_t
*mediated_cfg
, *mediation_cfg
;
76 enumerator_t
*enumerator
;
79 mediated_sa
= charon
->ike_sa_manager
->checkout(charon
->ike_sa_manager
,
80 this->mediated_sa_id
);
83 DBG1(DBG_IKE
, "initiating mediation connection");
84 mediated_cfg
= mediated_sa
->get_peer_cfg(mediated_sa
);
85 mediated_cfg
->get_ref(mediated_cfg
);
87 charon
->ike_sa_manager
->checkin(charon
->ike_sa_manager
, mediated_sa
);
89 mediation_cfg
= mediated_cfg
->get_mediated_by(mediated_cfg
);
90 mediation_cfg
->get_ref(mediation_cfg
);
92 enumerator
= mediation_cfg
->create_auth_cfg_enumerator(mediation_cfg
,
94 if (!enumerator
->enumerate(enumerator
, &auth_cfg
) ||
95 auth_cfg
->get(auth_cfg
, AUTH_RULE_IDENTITY
) == NULL
)
97 mediated_cfg
->destroy(mediated_cfg
);
98 mediation_cfg
->destroy(mediation_cfg
);
99 enumerator
->destroy(enumerator
);
103 enumerator
->destroy(enumerator
);
105 if (charon
->connect_manager
->check_and_register(charon
->connect_manager
,
106 auth_cfg
->get(auth_cfg
, AUTH_RULE_IDENTITY
),
107 mediated_cfg
->get_peer_id(mediated_cfg
),
108 this->mediated_sa_id
))
110 mediated_cfg
->destroy(mediated_cfg
);
111 mediation_cfg
->destroy(mediation_cfg
);
113 mediated_sa
= charon
->ike_sa_manager
->checkout(
114 charon
->ike_sa_manager
, this->mediated_sa_id
);
117 DBG1(DBG_IKE
, "mediation with the same peer is already in "
119 charon
->ike_sa_manager
->checkin(
120 charon
->ike_sa_manager
, mediated_sa
);
125 /* we need an additional reference because initiate consumes one */
126 mediation_cfg
->get_ref(mediation_cfg
);
128 if (charon
->controller
->initiate(charon
->controller
, mediation_cfg
,
129 NULL
, (controller_cb_t
)initiate_callback
, this) != SUCCESS
)
131 mediation_cfg
->destroy(mediation_cfg
);
132 mediated_cfg
->destroy(mediated_cfg
);
133 mediated_sa
= charon
->ike_sa_manager
->checkout(
134 charon
->ike_sa_manager
, this->mediated_sa_id
);
137 DBG1(DBG_IKE
, "initiating mediation connection failed");
138 charon
->ike_sa_manager
->checkin_and_destroy(
139 charon
->ike_sa_manager
, mediated_sa
);
144 mediation_cfg
->destroy(mediation_cfg
);
146 mediation_sa
= charon
->ike_sa_manager
->checkout(charon
->ike_sa_manager
,
147 this->mediation_sa_id
);
150 if (mediation_sa
->initiate_mediation(mediation_sa
,
151 mediated_cfg
) != SUCCESS
)
153 mediated_cfg
->destroy(mediated_cfg
);
154 charon
->ike_sa_manager
->checkin_and_destroy(
155 charon
->ike_sa_manager
, mediation_sa
);
156 mediated_sa
= charon
->ike_sa_manager
->checkout(
157 charon
->ike_sa_manager
, this->mediated_sa_id
);
160 DBG1(DBG_IKE
, "establishing mediation connection failed");
161 charon
->ike_sa_manager
->checkin_and_destroy(
162 charon
->ike_sa_manager
, mediated_sa
);
167 charon
->ike_sa_manager
->checkin(charon
->ike_sa_manager
,
170 mediated_cfg
->destroy(mediated_cfg
);
176 * Implementation of job_t.execute.
178 static void reinitiate(private_initiate_mediation_job_t
*this)
180 ike_sa_t
*mediated_sa
, *mediation_sa
;
181 peer_cfg_t
*mediated_cfg
;
183 mediated_sa
= charon
->ike_sa_manager
->checkout(charon
->ike_sa_manager
,
184 this->mediated_sa_id
);
187 mediated_cfg
= mediated_sa
->get_peer_cfg(mediated_sa
);
188 mediated_cfg
->get_ref(mediated_cfg
);
189 charon
->ike_sa_manager
->checkin(charon
->ike_sa_manager
, mediated_sa
);
191 mediation_sa
= charon
->ike_sa_manager
->checkout(charon
->ike_sa_manager
,
192 this->mediation_sa_id
);
195 if (mediation_sa
->initiate_mediation(mediation_sa
,
196 mediated_cfg
) != SUCCESS
)
198 DBG1(DBG_JOB
, "initiating mediated connection '%s' failed",
199 mediated_cfg
->get_name(mediated_cfg
));
200 mediated_cfg
->destroy(mediated_cfg
);
201 charon
->ike_sa_manager
->checkin_and_destroy(
202 charon
->ike_sa_manager
,
204 mediated_sa
= charon
->ike_sa_manager
->checkout(
205 charon
->ike_sa_manager
,
206 this->mediated_sa_id
);
209 DBG1(DBG_IKE
, "establishing mediation connection failed");
210 charon
->ike_sa_manager
->checkin_and_destroy(
211 charon
->ike_sa_manager
,
217 charon
->ike_sa_manager
->checkin(charon
->ike_sa_manager
,
221 mediated_cfg
->destroy(mediated_cfg
);
227 * Implementation of job_t.get_priority.
229 static job_priority_t
get_priority(private_initiate_mediation_job_t
*this)
231 return JOB_PRIO_MEDIUM
;
235 * Creates an empty job
237 static private_initiate_mediation_job_t
*initiate_mediation_job_create_empty()
239 private_initiate_mediation_job_t
*this = malloc_thing(private_initiate_mediation_job_t
);
241 /* interface functions */
242 this->public.job_interface
.get_priority
= (job_priority_t (*) (job_t
*)) get_priority
;
243 this->public.job_interface
.destroy
= (void (*) (job_t
*)) destroy
;
245 /* private variables */
246 this->mediation_sa_id
= NULL
;
247 this->mediated_sa_id
= NULL
;
253 * Described in header
255 initiate_mediation_job_t
*initiate_mediation_job_create(ike_sa_id_t
*ike_sa_id
)
257 private_initiate_mediation_job_t
*this = initiate_mediation_job_create_empty();
259 this->public.job_interface
.execute
= (void (*) (job_t
*)) initiate
;
261 this->mediated_sa_id
= ike_sa_id
->clone(ike_sa_id
);
263 return &this->public;
267 * Described in header
269 initiate_mediation_job_t
*reinitiate_mediation_job_create(ike_sa_id_t
*mediation_sa_id
,
270 ike_sa_id_t
*mediated_sa_id
)
272 private_initiate_mediation_job_t
*this = initiate_mediation_job_create_empty();
274 this->public.job_interface
.execute
= (void (*) (job_t
*)) reinitiate
;
276 this->mediation_sa_id
= mediation_sa_id
->clone(mediation_sa_id
);
277 this->mediated_sa_id
= mediated_sa_id
->clone(mediated_sa_id
);
279 return &this->public;