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
;
44 METHOD(job_t
, destroy
, void,
45 private_initiate_mediation_job_t
*this)
47 DESTROY_IF(this->mediation_sa_id
);
48 DESTROY_IF(this->mediated_sa_id
);
53 * Callback to handle initiation of mediation connection
55 static bool initiate_callback(private_initiate_mediation_job_t
*this,
56 debug_t group
, level_t level
, ike_sa_t
*ike_sa
,
59 if (ike_sa
&& !this->mediation_sa_id
)
61 this->mediation_sa_id
= ike_sa
->get_id(ike_sa
);
62 this->mediation_sa_id
= this->mediation_sa_id
->clone(this->mediation_sa_id
);
67 METHOD(job_t
, initiate
, job_requeue_t
,
68 private_initiate_mediation_job_t
*this)
70 ike_sa_t
*mediated_sa
, *mediation_sa
;
71 peer_cfg_t
*mediated_cfg
, *mediation_cfg
;
72 enumerator_t
*enumerator
;
75 mediated_sa
= charon
->ike_sa_manager
->checkout(charon
->ike_sa_manager
,
76 this->mediated_sa_id
);
79 DBG1(DBG_IKE
, "initiating mediation connection");
80 mediated_cfg
= mediated_sa
->get_peer_cfg(mediated_sa
);
81 mediated_cfg
->get_ref(mediated_cfg
);
83 charon
->ike_sa_manager
->checkin(charon
->ike_sa_manager
, mediated_sa
);
85 mediation_cfg
= mediated_cfg
->get_mediated_by(mediated_cfg
);
86 mediation_cfg
->get_ref(mediation_cfg
);
88 enumerator
= mediation_cfg
->create_auth_cfg_enumerator(mediation_cfg
,
90 if (!enumerator
->enumerate(enumerator
, &auth_cfg
) ||
91 auth_cfg
->get(auth_cfg
, AUTH_RULE_IDENTITY
) == NULL
)
93 mediated_cfg
->destroy(mediated_cfg
);
94 mediation_cfg
->destroy(mediation_cfg
);
95 enumerator
->destroy(enumerator
);
96 return JOB_REQUEUE_NONE
;
98 enumerator
->destroy(enumerator
);
100 if (charon
->connect_manager
->check_and_register(charon
->connect_manager
,
101 auth_cfg
->get(auth_cfg
, AUTH_RULE_IDENTITY
),
102 mediated_cfg
->get_peer_id(mediated_cfg
),
103 this->mediated_sa_id
))
105 mediated_cfg
->destroy(mediated_cfg
);
106 mediation_cfg
->destroy(mediation_cfg
);
108 mediated_sa
= charon
->ike_sa_manager
->checkout(
109 charon
->ike_sa_manager
, this->mediated_sa_id
);
112 DBG1(DBG_IKE
, "mediation with the same peer is already in "
114 charon
->ike_sa_manager
->checkin(
115 charon
->ike_sa_manager
, mediated_sa
);
117 return JOB_REQUEUE_NONE
;
119 /* we need an additional reference because initiate consumes one */
120 mediation_cfg
->get_ref(mediation_cfg
);
122 if (charon
->controller
->initiate(charon
->controller
, mediation_cfg
,
123 NULL
, (controller_cb_t
)initiate_callback
, this, 0) != SUCCESS
)
125 mediation_cfg
->destroy(mediation_cfg
);
126 mediated_cfg
->destroy(mediated_cfg
);
127 mediated_sa
= charon
->ike_sa_manager
->checkout(
128 charon
->ike_sa_manager
, this->mediated_sa_id
);
131 DBG1(DBG_IKE
, "initiating mediation connection failed");
132 charon
->ike_sa_manager
->checkin_and_destroy(
133 charon
->ike_sa_manager
, mediated_sa
);
135 return JOB_REQUEUE_NONE
;
137 mediation_cfg
->destroy(mediation_cfg
);
139 mediation_sa
= charon
->ike_sa_manager
->checkout(charon
->ike_sa_manager
,
140 this->mediation_sa_id
);
143 if (mediation_sa
->initiate_mediation(mediation_sa
,
144 mediated_cfg
) != SUCCESS
)
146 mediated_cfg
->destroy(mediated_cfg
);
147 charon
->ike_sa_manager
->checkin_and_destroy(
148 charon
->ike_sa_manager
, mediation_sa
);
149 mediated_sa
= charon
->ike_sa_manager
->checkout(
150 charon
->ike_sa_manager
, this->mediated_sa_id
);
153 DBG1(DBG_IKE
, "establishing mediation connection failed");
154 charon
->ike_sa_manager
->checkin_and_destroy(
155 charon
->ike_sa_manager
, mediated_sa
);
157 return JOB_REQUEUE_NONE
;
159 charon
->ike_sa_manager
->checkin(charon
->ike_sa_manager
,
162 mediated_cfg
->destroy(mediated_cfg
);
164 return JOB_REQUEUE_NONE
;
167 METHOD(job_t
, reinitiate
, job_requeue_t
,
168 private_initiate_mediation_job_t
*this)
170 ike_sa_t
*mediated_sa
, *mediation_sa
;
171 peer_cfg_t
*mediated_cfg
;
173 mediated_sa
= charon
->ike_sa_manager
->checkout(charon
->ike_sa_manager
,
174 this->mediated_sa_id
);
177 mediated_cfg
= mediated_sa
->get_peer_cfg(mediated_sa
);
178 mediated_cfg
->get_ref(mediated_cfg
);
179 charon
->ike_sa_manager
->checkin(charon
->ike_sa_manager
, mediated_sa
);
181 mediation_sa
= charon
->ike_sa_manager
->checkout(charon
->ike_sa_manager
,
182 this->mediation_sa_id
);
185 if (mediation_sa
->initiate_mediation(mediation_sa
,
186 mediated_cfg
) != SUCCESS
)
188 DBG1(DBG_JOB
, "initiating mediated connection '%s' failed",
189 mediated_cfg
->get_name(mediated_cfg
));
190 mediated_cfg
->destroy(mediated_cfg
);
191 charon
->ike_sa_manager
->checkin_and_destroy(
192 charon
->ike_sa_manager
,
194 mediated_sa
= charon
->ike_sa_manager
->checkout(
195 charon
->ike_sa_manager
,
196 this->mediated_sa_id
);
199 DBG1(DBG_IKE
, "establishing mediation connection failed");
200 charon
->ike_sa_manager
->checkin_and_destroy(
201 charon
->ike_sa_manager
,
204 return JOB_REQUEUE_NONE
;
206 charon
->ike_sa_manager
->checkin(charon
->ike_sa_manager
,
210 mediated_cfg
->destroy(mediated_cfg
);
212 return JOB_REQUEUE_NONE
;
215 METHOD(job_t
, get_priority
, job_priority_t
,
216 private_initiate_mediation_job_t
*this)
218 return JOB_PRIO_MEDIUM
;
222 * Creates an empty job
224 static private_initiate_mediation_job_t
*initiate_mediation_job_create_empty()
226 private_initiate_mediation_job_t
*this;
230 .get_priority
= _get_priority
,
239 * Described in header
241 initiate_mediation_job_t
*initiate_mediation_job_create(ike_sa_id_t
*ike_sa_id
)
243 private_initiate_mediation_job_t
*this = initiate_mediation_job_create_empty();
245 this->public.job_interface
.execute
= _initiate
;
246 this->mediated_sa_id
= ike_sa_id
->clone(ike_sa_id
);
248 return &this->public;
252 * Described in header
254 initiate_mediation_job_t
*reinitiate_mediation_job_create(ike_sa_id_t
*mediation_sa_id
,
255 ike_sa_id_t
*mediated_sa_id
)
257 private_initiate_mediation_job_t
*this = initiate_mediation_job_create_empty();
259 this->public.job_interface
.execute
= _reinitiate
;
260 this->mediation_sa_id
= mediation_sa_id
->clone(mediation_sa_id
);
261 this->mediated_sa_id
= mediated_sa_id
->clone(mediated_sa_id
);
263 return &this->public;