#include <stdlib.h>
#include <sys/types.h>
#include <dhcpmsg.h>
#include <dhcpagent_ipc.h>
#include "agent.h"
#include "states.h"
#include "interface.h"
#include "ipc_action.h"
#include "util.h"
static iu_tq_callback_t ipc_action_timeout;
void
ipc_action_init(ipc_action_t *ia)
{
ia->ia_cmd = 0;
ia->ia_fd = -1;
ia->ia_tid = -1;
ia->ia_eid = -1;
ia->ia_request = NULL;
}
boolean_t
ipc_action_start(dhcp_smach_t *dsmp, ipc_action_t *iareq)
{
struct ipc_action *ia = &dsmp->dsm_ia;
if (ia->ia_fd != -1 || ia->ia_tid != -1 || iareq->ia_fd == -1) {
dhcpmsg(MSG_CRIT, "ipc_action_start: attempted restart on %s",
dsmp->dsm_name);
return (B_FALSE);
}
if (!async_cancel(dsmp)) {
dhcpmsg(MSG_WARNING, "ipc_action_start: unable to cancel "
"action on %s", dsmp->dsm_name);
return (B_FALSE);
}
if (iareq->ia_request->timeout == DHCP_IPC_WAIT_DEFAULT)
iareq->ia_request->timeout = DHCP_IPC_DEFAULT_WAIT;
if (iareq->ia_request->timeout == DHCP_IPC_WAIT_FOREVER) {
iareq->ia_tid = -1;
} else {
iareq->ia_tid = iu_schedule_timer(tq,
iareq->ia_request->timeout, ipc_action_timeout, dsmp);
if (iareq->ia_tid == -1) {
dhcpmsg(MSG_ERROR, "ipc_action_start: failed to set "
"timer for %s on %s",
dhcp_ipc_type_to_string(iareq->ia_cmd),
dsmp->dsm_name);
return (B_FALSE);
}
hold_smach(dsmp);
}
*ia = *iareq;
ipc_action_init(iareq);
dhcpmsg(MSG_DEBUG, "ipc_action_start: started %s (command %d) on %s,"
" buffer length %u",
dhcp_ipc_type_to_string(ia->ia_cmd), ia->ia_cmd, dsmp->dsm_name,
ia->ia_request == NULL ? 0 : ia->ia_request->data_length);
dsmp->dsm_dflags |= DHCP_IF_BUSY;
(void) async_start(dsmp, ia->ia_cmd, B_TRUE);
return (B_TRUE);
}
void
ipc_action_finish(dhcp_smach_t *dsmp, int reason)
{
struct ipc_action *ia = &dsmp->dsm_ia;
dsmp->dsm_dflags &= ~DHCP_IF_BUSY;
if (dsmp->dsm_ia.ia_fd == -1) {
dhcpmsg(MSG_ERROR,
"ipc_action_finish: attempted to finish unknown action "
"on %s", dsmp->dsm_name);
return;
}
dhcpmsg(MSG_DEBUG,
"ipc_action_finish: finished %s (command %d) on %s: %d",
dhcp_ipc_type_to_string(ia->ia_cmd), (int)ia->ia_cmd,
dsmp->dsm_name, reason);
if (dsmp->dsm_ia.ia_tid != -1 &&
iu_cancel_timer(tq, dsmp->dsm_ia.ia_tid, NULL) == 1) {
dsmp->dsm_ia.ia_tid = -1;
release_smach(dsmp);
}
if (reason == 0)
send_ok_reply(ia);
else
send_error_reply(ia, reason);
async_finish(dsmp);
}
static void
ipc_action_timeout(iu_tq_t *tq, void *arg)
{
dhcp_smach_t *dsmp = arg;
struct ipc_action *ia = &dsmp->dsm_ia;
dsmp->dsm_dflags &= ~DHCP_IF_BUSY;
ia->ia_tid = -1;
dhcpmsg(MSG_VERBOSE, "ipc timeout waiting for agent to complete "
"%s (command %d) for %s", dhcp_ipc_type_to_string(ia->ia_cmd),
ia->ia_cmd, dsmp->dsm_name);
send_error_reply(ia, DHCP_IPC_E_TIMEOUT);
async_finish(dsmp);
release_smach(dsmp);
}
void
send_ok_reply(ipc_action_t *ia)
{
send_error_reply(ia, 0);
}
void
send_error_reply(ipc_action_t *ia, int error)
{
send_data_reply(ia, error, DHCP_TYPE_NONE, NULL, 0);
}
void
send_data_reply(ipc_action_t *ia, int error, dhcp_data_type_t type,
const void *buffer, size_t size)
{
dhcp_ipc_reply_t *reply;
int retval;
if (ia->ia_fd == -1 || ia->ia_request == NULL)
return;
reply = dhcp_ipc_alloc_reply(ia->ia_request, error, buffer, size,
type);
if (reply == NULL) {
dhcpmsg(MSG_ERR, "send_data_reply: cannot allocate reply");
} else if ((retval = dhcp_ipc_send_reply(ia->ia_fd, reply)) != 0) {
dhcpmsg(MSG_ERROR, "send_data_reply: dhcp_ipc_send_reply: %s",
dhcp_ipc_strerror(retval));
}
free(reply);
free(ia->ia_request);
if (ia->ia_eid != -1)
(void) iu_unregister_event(eh, ia->ia_eid, NULL);
(void) dhcp_ipc_close(ia->ia_fd);
ia->ia_request = NULL;
ia->ia_fd = -1;
ia->ia_eid = -1;
}