aboutsummaryrefslogtreecommitdiffstats
path: root/src
diff options
context:
space:
mode:
Diffstat (limited to 'src')
-rwxr-xr-x[-rw-r--r--]src/libcharon/Makefile.am1
-rwxr-xr-x[-rw-r--r--]src/libcharon/sa/ikev1/task_manager_v1.c85
-rwxr-xr-x[-rw-r--r--]src/libcharon/sa/ikev1/task_manager_v1.h5
-rwxr-xr-x[-rw-r--r--]src/libcharon/sa/ikev1/tasks/aggressive_mode.c2
-rwxr-xr-x[-rw-r--r--]src/libcharon/sa/ikev1/tasks/informational.c36
-rwxr-xr-x[-rw-r--r--]src/libcharon/sa/ikev1/tasks/informational.h3
-rwxr-xr-xsrc/libcharon/sa/ikev1/tasks/isakmp_dpd.c278
-rwxr-xr-xsrc/libcharon/sa/ikev1/tasks/isakmp_dpd.h38
-rwxr-xr-x[-rw-r--r--]src/libcharon/sa/ikev1/tasks/main_mode.c2
-rwxr-xr-x[-rw-r--r--]src/libcharon/sa/ikev1/tasks/quick_mode.c2
-rwxr-xr-x[-rw-r--r--]src/libcharon/sa/task.h3
11 files changed, 446 insertions, 9 deletions
diff --git a/src/libcharon/Makefile.am b/src/libcharon/Makefile.am
index c85c472b8..d5e139412 100644..100755
--- a/src/libcharon/Makefile.am
+++ b/src/libcharon/Makefile.am
@@ -113,6 +113,7 @@ sa/ikev1/tasks/isakmp_cert_post.c sa/ikev1/tasks/isakmp_cert_post.h \
sa/ikev1/tasks/isakmp_natd.c sa/ikev1/tasks/isakmp_natd.h \
sa/ikev1/tasks/isakmp_vendor.c sa/ikev1/tasks/isakmp_vendor.h \
sa/ikev1/tasks/isakmp_delete.c sa/ikev1/tasks/isakmp_delete.h \
+sa/ikev1/tasks/isakmp_dpd.c sa/ikev1/tasks/isakmp_dpd.h \
sa/ikev1/tasks/xauth.c sa/ikev1/tasks/xauth.h \
sa/ikev1/tasks/quick_mode.c sa/ikev1/tasks/quick_mode.h \
sa/ikev1/tasks/quick_delete.c sa/ikev1/tasks/quick_delete.h \
diff --git a/src/libcharon/sa/ikev1/task_manager_v1.c b/src/libcharon/sa/ikev1/task_manager_v1.c
index f7415a5ad..7e1ff616a 100644..100755
--- a/src/libcharon/sa/ikev1/task_manager_v1.c
+++ b/src/libcharon/sa/ikev1/task_manager_v1.c
@@ -31,6 +31,8 @@
#include <sa/ikev1/tasks/isakmp_cert_pre.h>
#include <sa/ikev1/tasks/isakmp_cert_post.h>
#include <sa/ikev1/tasks/isakmp_delete.h>
+#include <sa/ikev1/tasks/isakmp_dpd.h>
+
#include <processing/jobs/retransmit_job.h>
#include <processing/jobs/delete_ike_sa_job.h>
@@ -197,6 +199,16 @@ struct private_task_manager_t {
* Base to calculate retransmission timeout
*/
double retransmit_base;
+
+ /**
+ * Sequence number for sending DPD requests
+ */
+ u_int32_t dpd_send_seqnr;
+
+ /**
+ * Sequence number for received DPD requests
+ */
+ u_int32_t dpd_rec_seqnr;
};
/**
@@ -406,6 +418,13 @@ METHOD(task_manager_t, initiate, status_t,
new_mid = TRUE;
break;
}
+
+ if (activate_task(this, TASK_ISAKMP_DPD))
+ {
+ exchange = INFORMATIONAL_V1;
+ new_mid = TRUE;
+ break;
+ }
break;
default:
break;
@@ -580,7 +599,20 @@ static status_t build_response(private_task_manager_t *this, message_t *request)
/* send response along the path the request came in */
message->set_source(message, me->clone(me));
message->set_destination(message, other->clone(other));
- message->set_message_id(message, request->get_message_id(request));
+
+ /* Create new message id for informational exchanges*/
+ if (request->get_exchange_type(request) == INFORMATIONAL_V1)
+ {
+ u_int32_t message_id;
+
+ this->rng->get_bytes(this->rng, sizeof(message_id),
+ (void*)&message_id);
+ message->set_message_id(message, message_id);
+ }
+ else
+ {
+ message->set_message_id(message, request->get_message_id(request));
+ }
message->set_request(message, FALSE);
this->responding.mid = request->get_message_id(request);
@@ -710,6 +742,7 @@ static status_t process_request(private_task_manager_t *this,
enumerator_t *enumerator;
task_t *task = NULL;
bool send_response = FALSE;
+ bool informational = FALSE;
if (message->get_exchange_type(message) == INFORMATIONAL_V1 ||
this->passive_tasks->get_count(this->passive_tasks) == 0)
@@ -752,8 +785,9 @@ static status_t process_request(private_task_manager_t *this,
this->passive_tasks->insert_last(this->passive_tasks, task);
break;
case INFORMATIONAL_V1:
- task = (task_t *)informational_create(this->ike_sa, NULL);
+ task = (task_t *)informational_create(this->ike_sa, NULL, this->dpd_rec_seqnr);
this->passive_tasks->insert_first(this->passive_tasks, task);
+ informational = TRUE;
break;
case TRANSACTION:
if (this->ike_sa->get_state(this->ike_sa) == IKE_ESTABLISHED)
@@ -784,6 +818,15 @@ static status_t process_request(private_task_manager_t *this,
case NEED_MORE:
/* processed, but task needs at least another call to build() */
send_response = TRUE;
+ if (informational && !this->dpd_rec_seqnr)
+ {
+ /* Update the received DPD sequence number if it the first received one */
+ if (task->get_type(task) == TASK_ISAKMP_DPD)
+ {
+ isakmp_dpd_t *isakmp_dpd = (isakmp_dpd_t *)task;
+ this->dpd_rec_seqnr = isakmp_dpd->get_dpd_seqnr(isakmp_dpd);
+ }
+ }
continue;
case ALREADY_DONE:
send_response = FALSE;
@@ -950,6 +993,7 @@ METHOD(task_manager_t, process_message, status_t,
u_int32_t hash, mid, i;
host_t *me, *other;
status_t status;
+ bool dpd_response = FALSE;
/* TODO-IKEv1: update hosts more selectively */
me = msg->get_destination(msg);
@@ -977,7 +1021,27 @@ METHOD(task_manager_t, process_message, status_t,
}
}
- if ((mid && mid == this->initiating.mid) ||
+ /* DPD Acks are not sent with a same message ID as the request.*/
+ if (msg->get_exchange_type(msg) == INFORMATIONAL_V1 &&
+ this->active_tasks->get_count(this->active_tasks))
+ {
+ enumerator_t *enumerator;
+ task_t *task;
+ /* In case of ongoing DPD request, let the DPD task handle all information exchanges. */
+ enumerator = this->active_tasks->create_enumerator(this->active_tasks);
+ while (enumerator->enumerate(enumerator, (void**)&task))
+ {
+ if (task->get_type(task) == TASK_ISAKMP_DPD)
+ {
+ dpd_response = TRUE;
+ break;
+ }
+ }
+ enumerator->destroy(enumerator);
+ }
+
+
+ if ((mid && mid == this->initiating.mid) || dpd_response ||
(this->initiating.mid == 0 &&
msg->get_exchange_type(msg) == this->initiating.type &&
this->active_tasks->get_count(this->active_tasks)))
@@ -1271,10 +1335,16 @@ METHOD(task_manager_t, queue_child_delete, void,
spi, FALSE, expired));
}
+METHOD(task_manager_v1_t, get_dpd_seqnr, u_int32_t,
+ private_task_manager_t *this)
+{
+ return this->dpd_send_seqnr++;
+}
+
METHOD(task_manager_t, queue_dpd, void,
private_task_manager_t *this)
{
- /* TODO-IKEv1: DPD checking */
+ queue_task(this, (task_t*)isakmp_dpd_create(this->ike_sa, NULL, _get_dpd_seqnr(&this->public)));
}
METHOD(task_manager_t, adopt_tasks, void,
@@ -1401,6 +1471,7 @@ task_manager_v1_t *task_manager_v1_create(ike_sa_t *ike_sa)
.create_task_enumerator = _create_task_enumerator,
.destroy = _destroy,
},
+ .get_dpd_seqnr = _get_dpd_seqnr,
},
.ike_sa = ike_sa,
.initiating.type = EXCHANGE_TYPE_UNDEFINED,
@@ -1417,5 +1488,11 @@ task_manager_v1_t *task_manager_v1_create(ike_sa_t *ike_sa)
"charon.retransmit_base", RETRANSMIT_BASE),
);
+ this->rng->get_bytes(this->rng, sizeof(this->dpd_send_seqnr),
+ (void*)&this->dpd_send_seqnr);
+
+ this->dpd_send_seqnr &= 0x7FFFFFFF;
+
return &this->public;
}
+
diff --git a/src/libcharon/sa/ikev1/task_manager_v1.h b/src/libcharon/sa/ikev1/task_manager_v1.h
index 99cd35e32..58b20d0c2 100644..100755
--- a/src/libcharon/sa/ikev1/task_manager_v1.h
+++ b/src/libcharon/sa/ikev1/task_manager_v1.h
@@ -34,6 +34,11 @@ struct task_manager_v1_t {
* Implements task_manager_t.
*/
task_manager_t task_manager;
+
+ /**
+ * Get new sequence number to use for sending DPD request.
+ */
+ u_int32_t (*get_dpd_seqnr) (task_manager_v1_t *this);
};
/**
diff --git a/src/libcharon/sa/ikev1/tasks/aggressive_mode.c b/src/libcharon/sa/ikev1/tasks/aggressive_mode.c
index 1fe36a9d5..df0c94b28 100644..100755
--- a/src/libcharon/sa/ikev1/tasks/aggressive_mode.c
+++ b/src/libcharon/sa/ikev1/tasks/aggressive_mode.c
@@ -164,7 +164,7 @@ static status_t send_notify(private_aggressive_mode_t *this, notify_type_t type)
notify->set_spi_data(notify, spi);
this->ike_sa->queue_task(this->ike_sa,
- (task_t*)informational_create(this->ike_sa, notify));
+ (task_t*)informational_create(this->ike_sa, notify, 0));
/* cancel all active/passive tasks in favour of informational */
return ALREADY_DONE;
}
diff --git a/src/libcharon/sa/ikev1/tasks/informational.c b/src/libcharon/sa/ikev1/tasks/informational.c
index 999b497dc..2af641e86 100644..100755
--- a/src/libcharon/sa/ikev1/tasks/informational.c
+++ b/src/libcharon/sa/ikev1/tasks/informational.c
@@ -18,6 +18,8 @@
#include <daemon.h>
#include <sa/ikev1/tasks/isakmp_delete.h>
#include <sa/ikev1/tasks/quick_delete.h>
+#include <sa/ikev1/tasks/isakmp_dpd.h>
+
#include <encoding/payloads/delete_payload.h>
typedef struct private_informational_t private_informational_t;
@@ -46,6 +48,16 @@ struct private_informational_t {
* Delete subtask
*/
task_t *del;
+
+ /**
+ * DPD subtask
+ */
+ task_t *dpd;
+
+ /**
+ * DPD sequence number
+ */
+ u_int32_t dpd_seqnr;
};
METHOD(task_t, build_i, status_t,
@@ -80,6 +92,15 @@ METHOD(task_t, process_r, status_t,
this->ike_sa->set_condition(this->ike_sa,
COND_INIT_CONTACT_SEEN, TRUE);
}
+ else if (type == DPD_R_U_THERE)
+ {
+ DBG3(DBG_IKE, "received DPD request");
+ this->dpd = (task_t*)isakmp_dpd_create(this->ike_sa, notify, this->dpd_seqnr);
+ }
+ else if (type == DPD_R_U_THERE_ACK)
+ {
+ DBG3(DBG_IKE, "received DPD request ack");
+ }
else if (type < 16384)
{
DBG1(DBG_IKE, "received %N error notify",
@@ -123,6 +144,11 @@ METHOD(task_t, process_r, status_t,
{
return this->del->process(this->del, message);
}
+
+ if (this->dpd && status == SUCCESS)
+ {
+ return this->dpd->process(this->dpd, message);
+ }
return status;
}
@@ -133,6 +159,13 @@ METHOD(task_t, build_r, status_t,
{
return this->del->build(this->del, message);
}
+
+ if (this->dpd)
+ {
+ status_t status = this->dpd->build(this->dpd, message);
+ this->dpd->destroy(this->dpd);
+ return status;
+ }
return FAILED;
}
@@ -165,7 +198,7 @@ METHOD(task_t, destroy, void,
/*
* Described in header.
*/
-informational_t *informational_create(ike_sa_t *ike_sa, notify_payload_t *notify)
+informational_t *informational_create(ike_sa_t *ike_sa, notify_payload_t *notify, u_int32_t dpd_seqnr)
{
private_informational_t *this;
@@ -179,6 +212,7 @@ informational_t *informational_create(ike_sa_t *ike_sa, notify_payload_t *notify
},
.ike_sa = ike_sa,
.notify = notify,
+ .dpd_seqnr = dpd_seqnr,
);
if (notify)
diff --git a/src/libcharon/sa/ikev1/tasks/informational.h b/src/libcharon/sa/ikev1/tasks/informational.h
index f1543dc58..59b811000 100644..100755
--- a/src/libcharon/sa/ikev1/tasks/informational.h
+++ b/src/libcharon/sa/ikev1/tasks/informational.h
@@ -44,8 +44,9 @@ struct informational_t {
*
* @param ike_sa IKE_SA this task works for
* @param notify notify to send as initiator, NULL if responder
+ * @param dpd_seqnr DPD sequence number, incoming or outgoing
* @return task to handle by the task_manager
*/
-informational_t *informational_create(ike_sa_t *ike_sa, notify_payload_t *notify);
+informational_t *informational_create(ike_sa_t *ike_sa, notify_payload_t *notify, u_int32_t dpd_seqnr);
#endif /** INFORMATIONAL_H_ @}*/
diff --git a/src/libcharon/sa/ikev1/tasks/isakmp_dpd.c b/src/libcharon/sa/ikev1/tasks/isakmp_dpd.c
new file mode 100755
index 000000000..ec0a1eddc
--- /dev/null
+++ b/src/libcharon/sa/ikev1/tasks/isakmp_dpd.c
@@ -0,0 +1,278 @@
+#include "isakmp_dpd.h"
+
+#include <encoding/payloads/notify_payload.h>
+#include <sa/ikev1/tasks/informational.h>
+
+#include <daemon.h>
+
+#ifdef SLIPSTREAM
+/* Should be the last include */
+#include <ikev2_mem.h>
+#endif /* SLIPSTREAM */
+
+typedef struct private_isakmp_dpd_t private_isakmp_dpd_t;
+
+/**
+ * Private members of a isakmp_dpd_t task.
+ */
+struct private_isakmp_dpd_t {
+
+ /**
+ * Public methods and task_t interface.
+ */
+ isakmp_dpd_t public;
+
+ /**
+ * Sequence number.
+ */
+ u_int32_t seqnr;
+
+ /**
+ * Notify payload, only provided for requests.
+ */
+ notify_payload_t *notify;
+
+ /**
+ * IKE SA we are serving.
+ */
+ ike_sa_t *ike_sa;
+};
+
+/**
+ * Get DPD sequence number from notify payload.
+ */
+static bool get_seqnr(notify_payload_t *notify, u_int32_t *seqnr)
+{
+ chunk_t chunk = notify->get_notification_data(notify);
+
+ if( chunk.ptr && chunk.len == 4)
+ {
+ u_int32_t seqnr_read = *((u_int32_t*)chunk.ptr);
+
+ *seqnr = ntohl(seqnr_read);
+
+ return TRUE;
+ }
+
+ DBG1(DBG_IKE, "no DPD seqnr received");
+
+ return FALSE;
+}
+
+/**
+ * Add notify payload to message.
+ */
+static void add_notify(private_isakmp_dpd_t *this, message_t *message, notify_type_t type)
+{
+ notify_payload_t *notify;
+
+ ike_sa_id_t *ike_sa_id;
+ u_int64_t spi_i, spi_r;
+ u_int32_t seqnr;
+ chunk_t spi;
+
+ notify = notify_payload_create_from_protocol_and_type(NOTIFY_V1,
+ PROTO_IKE, type);
+
+ seqnr = htonl(this->seqnr);
+ ike_sa_id = this->ike_sa->get_id(this->ike_sa);
+ spi_i = ike_sa_id->get_initiator_spi(ike_sa_id);
+ spi_r = ike_sa_id->get_responder_spi(ike_sa_id);
+ spi = chunk_cata("cc", chunk_from_thing(spi_i), chunk_from_thing(spi_r));
+
+ notify->set_spi_data(notify, spi);
+ notify->set_notification_data(notify, chunk_from_thing(seqnr));
+
+ message->add_payload(message, (payload_t*)notify);
+}
+
+METHOD(isakmp_dpd_t, get_dpd_seqnr, u_int32_t,
+ private_isakmp_dpd_t *this)
+{
+ return this->seqnr;
+}
+
+METHOD(task_t, build_i, status_t,
+ private_isakmp_dpd_t *this, message_t *message)
+{
+ add_notify(this, message, DPD_R_U_THERE);
+
+ return NEED_MORE;
+}
+
+METHOD(task_t, build_r, status_t,
+ private_isakmp_dpd_t *this, message_t *message)
+{
+ add_notify(this, message, DPD_R_U_THERE_ACK);
+
+ return SUCCESS;
+}
+
+METHOD(task_t, process_i, status_t,
+ private_isakmp_dpd_t *this, message_t *message)
+{
+ enumerator_t *enumerator;
+ notify_payload_t *notify;
+ notify_type_t type;
+ payload_t *payload;
+ task_t *info_task = NULL;
+
+ enumerator = message->create_payload_enumerator(message);
+ while (enumerator->enumerate(enumerator, &payload))
+ {
+ switch (payload->get_type(payload))
+ {
+ case NOTIFY_V1:
+ notify = (notify_payload_t*)payload;
+ type = notify->get_notify_type(notify);
+
+ if (type == DPD_R_U_THERE_ACK)
+ {
+ u_int32_t seqnr;
+
+ if (!get_seqnr(notify, &seqnr))
+ {
+ return FAILED;
+ }
+
+ if (this->seqnr != seqnr)
+ {
+ DBG1(DBG_IKE, "received DPD Ack with unexpected seqnr (%u) expect (%u)",seqnr,this->seqnr);
+ return SUCCESS;
+ }
+
+ DBG4(DBG_IKE, "received DPD Ack with seqnr (%u)",seqnr);
+
+ return SUCCESS;
+
+ }
+ else if (type == DPD_R_U_THERE)
+ {
+ u_int32_t expected = this->seqnr + 1;
+
+ if (!get_seqnr(notify, &this->seqnr))
+ {
+ return FAILED;
+ }
+
+ if (expected != 1 && this->seqnr != expected)
+ {
+ DBG1(DBG_IKE, "received DPD request with unexpected seqnr (%u) expect (%u)",
+ this->seqnr,expected);
+ return SUCCESS;
+ }
+
+ DBG4(DBG_IKE, "received DPD request with seqnr %u",this->seqnr);
+
+ this->public.task.build = _build_r;
+ return NEED_MORE;
+ }
+ else
+ {
+ info_task = (task_t*)informational_create(this->ike_sa, NULL, 0);
+ }
+ continue;
+
+ default:
+ continue;
+ }
+ break;
+ }
+ enumerator->destroy(enumerator);
+
+ if (info_task)
+ {
+ status_t status = info_task->process(info_task, message);
+ /* Assuming that the informational task will not need to send other replies than dpd */
+ info_task->destroy(info_task);
+ return status;
+ }
+
+ return SUCCESS;
+}
+
+METHOD(task_t, process_r, status_t,
+ private_isakmp_dpd_t *this, message_t *message)
+{
+ u_int32_t expected = this->seqnr + 1;
+
+ if (this->notify)
+ {
+ if (!get_seqnr(this->notify, &this->seqnr))
+ {
+ return FAILED;
+ }
+
+ if (expected != 1 && this->seqnr != expected)
+ {
+ DBG1(DBG_IKE, "received DPD request with unexpected seqnr (%u) expect (%u)",
+ this->seqnr,expected);
+ return SUCCESS;
+ }
+
+ DBG4(DBG_IKE, "DPD request received with seqnr %u",this->seqnr);
+ }
+ else
+ {
+ DBG1(DBG_IKE, "no notify provided");
+ return FAILED;
+ }
+ return NEED_MORE;
+}
+
+
+METHOD(task_t, get_type, task_type_t,
+ private_isakmp_dpd_t *this)
+{
+ return TASK_ISAKMP_DPD;
+}
+
+
+METHOD(task_t, migrate, void,
+ private_isakmp_dpd_t *this, ike_sa_t *ike_sa)
+{
+ this->ike_sa = ike_sa;
+ this->seqnr = 0;
+
+}
+
+METHOD(task_t, destroy, void,
+ private_isakmp_dpd_t *this)
+{
+ free(this);
+}
+
+/*
+ * Described in header.
+ */
+isakmp_dpd_t *isakmp_dpd_create(ike_sa_t *ike_sa, notify_payload_t *notify, u_int32_t seqnr)
+{
+ private_isakmp_dpd_t *this;
+
+ INIT(this,
+ .public = {
+ .task = {
+ .get_type = _get_type,
+ .migrate = _migrate,
+ .destroy = _destroy,
+ },
+ .get_dpd_seqnr = _get_dpd_seqnr,
+ },
+ .notify = notify,
+ .ike_sa = ike_sa,
+ .seqnr = seqnr,
+ );
+
+ if (!notify)
+ {
+ this->public.task.build = _build_i;
+ this->public.task.process = _process_i;
+ }
+ else
+ {
+ this->public.task.build = _build_r;
+ this->public.task.process = _process_r;
+ }
+
+ return &this->public;
+}
diff --git a/src/libcharon/sa/ikev1/tasks/isakmp_dpd.h b/src/libcharon/sa/ikev1/tasks/isakmp_dpd.h
new file mode 100755
index 000000000..5d940b8d4
--- /dev/null
+++ b/src/libcharon/sa/ikev1/tasks/isakmp_dpd.h
@@ -0,0 +1,38 @@
+#ifndef ISAKMP_DPD_H_
+#define ISAPMP_DPD_H_
+
+typedef struct isakmp_dpd_t isakmp_dpd_t;
+
+#include <library.h>
+#include <sa/ike_sa.h>
+#include <sa/task.h>
+
+/**
+ * Task of type isakmp_dpd, detects dead peers.
+ *
+ *
+ */
+struct isakmp_dpd_t {
+
+ /**
+ * Implements the task_t interface
+ */
+ task_t task;
+
+ /**
+ * Get the received dpd seqnr.
+ *
+ * @return protocol ID
+ */
+ u_int32_t (*get_dpd_seqnr) (isakmp_dpd_t *dpd_task);
+};
+
+/**
+ * Create a new isakmp_dpd task.
+ *
+ * @param initiator TRUE if task is the original initiator
+ * @return isakmp_dpd task to handle by the task_manager
+ */
+isakmp_dpd_t *isakmp_dpd_create(ike_sa_t *ike_sa, notify_payload_t *notify, u_int32_t seqnr);
+
+#endif /** ISAKMP_DPD_H_ @}*/ \ No newline at end of file
diff --git a/src/libcharon/sa/ikev1/tasks/main_mode.c b/src/libcharon/sa/ikev1/tasks/main_mode.c
index ba1a9ad1d..f1e3290a7 100644..100755
--- a/src/libcharon/sa/ikev1/tasks/main_mode.c
+++ b/src/libcharon/sa/ikev1/tasks/main_mode.c
@@ -176,7 +176,7 @@ static status_t send_notify(private_main_mode_t *this, notify_type_t type)
notify->set_spi_data(notify, spi);
this->ike_sa->queue_task(this->ike_sa,
- (task_t*)informational_create(this->ike_sa, notify));
+ (task_t*)informational_create(this->ike_sa, notify, 0));
/* cancel all active/passive tasks in favour of informational */
return ALREADY_DONE;
}
diff --git a/src/libcharon/sa/ikev1/tasks/quick_mode.c b/src/libcharon/sa/ikev1/tasks/quick_mode.c
index dedeab189..b27b00de7 100644..100755
--- a/src/libcharon/sa/ikev1/tasks/quick_mode.c
+++ b/src/libcharon/sa/ikev1/tasks/quick_mode.c
@@ -539,7 +539,7 @@ static status_t send_notify(private_quick_mode_t *this, notify_type_t type)
notify->set_spi(notify, this->spi_i);
this->ike_sa->queue_task(this->ike_sa,
- (task_t*)informational_create(this->ike_sa, notify));
+ (task_t*)informational_create(this->ike_sa, notify, 0));
/* cancel all active/passive tasks in favour of informational */
return ALREADY_DONE;
}
diff --git a/src/libcharon/sa/task.h b/src/libcharon/sa/task.h
index 6b5c92930..1b42bf1a4 100644..100755
--- a/src/libcharon/sa/task.h
+++ b/src/libcharon/sa/task.h
@@ -93,6 +93,9 @@ enum task_type_t {
TASK_ISAKMP_CERT_PRE,
/** IKEv1 post-authentication certificate handling */
TASK_ISAKMP_CERT_POST,
+ /** IKEv1 DPD */
+ TASK_ISAKMP_DPD,
+
};
/**