/*
* Copyright 2009 Sun Microsystems, Inc. All rights reserved.
* Use is subject to license terms.
*/
/*
* BSD 3 Clause License
*
* Copyright (c) 2007, The Storage Networking Industry Association.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* - Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
*
* - Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in
* the documentation and/or other materials provided with the
* distribution.
*
* - Neither the name of The Storage Networking Industry Association (SNIA)
* nor the names of its contributors may be used to endorse or promote
* products derived from this software without specific prior written
* permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGE.
*/
/* This file contains all the door server code */
#include <door.h>
#include <alloca.h>
#include <errno.h>
#include <note.h>
#include <libintl.h>
#include <ndmpd_door.h>
#include "ndmpd.h"
/* static variables */
static int ndmp_door_fildes = -1;
static mutex_t ndmp_doorsrv_mutex;
/* static routines */
static void ndmp_door_server(void *cookie, char *ptr, size_t size,
door_desc_t *dp, uint_t n_desc);
/*
* Statistics used in ndmpstat command
*/
ndmp_stat_t ndstat;
int
ndmp_door_init(void)
{
int fd;
(void) mutex_lock(&ndmp_doorsrv_mutex);
if (ndmp_door_fildes != -1) {
NDMP_LOG(LOG_DEBUG,
"ndmp_door_init: ndmpd service is already running.");
(void) mutex_unlock(&ndmp_doorsrv_mutex);
return (0);
}
if ((ndmp_door_fildes = door_create(ndmp_door_server,
NULL, DOOR_UNREF)) < 0) {
NDMP_LOG(LOG_DEBUG, "ndmp_door_init: Could not create door.");
(void) mutex_unlock(&ndmp_doorsrv_mutex);
return (-1);
}
(void) unlink(NDMP_DOOR_SVC);
if ((fd = creat(NDMP_DOOR_SVC, 0444)) < 0) {
NDMP_LOG(LOG_DEBUG, "ndmp_door_init: Can't create %s: %m.",
NDMP_DOOR_SVC);
(void) door_revoke(ndmp_door_fildes);
ndmp_door_fildes = -1;
(void) mutex_unlock(&ndmp_doorsrv_mutex);
return (-1);
}
(void) close(fd);
(void) fdetach(NDMP_DOOR_SVC);
if (fattach(ndmp_door_fildes, NDMP_DOOR_SVC) < 0) {
NDMP_LOG(LOG_DEBUG, "ndmp_door_init: fattach failed %m");
(void) door_revoke(ndmp_door_fildes);
ndmp_door_fildes = -1;
(void) mutex_unlock(&ndmp_doorsrv_mutex);
return (-1);
}
NDMP_LOG(LOG_DEBUG, "ndmp_door_init: Door server successfully started");
(void) mutex_unlock(&ndmp_doorsrv_mutex);
return (0);
}
void
ndmp_door_fini(void)
{
(void) mutex_lock(&ndmp_doorsrv_mutex);
if (ndmp_door_fildes != -1) {
(void) fdetach(NDMP_DOOR_SVC);
(void) door_revoke(ndmp_door_fildes);
ndmp_door_fildes = -1;
}
(void) mutex_unlock(&ndmp_doorsrv_mutex);
}
boolean_t
ndmp_door_check(void)
{
door_info_t info;
int door;
if ((door = open(NDMP_DOOR_SVC, O_RDONLY)) < 0)
return (0);
if (door_info(door, &info) < 0) {
(void) close(door);
return (0);
}
if (info.di_target > 0) {
NDMP_LOG(LOG_ERR,
"Service already running: pid %ld", info.di_target);
(void) close(door);
return (1);
}
(void) close(door);
return (0);
}
/* door server */
/*ARGSUSED*/
void
ndmp_door_server(void *cookie, char *ptr, size_t size,
door_desc_t *dp, uint_t n_desc)
{
NOTE(ARGUNUSED(cookie,dp,n_desc))
int req_type;
char *buf;
int buflen;
unsigned int used;
ndmp_door_ctx_t *dec_ctx;
ndmp_door_ctx_t *enc_ctx;
unsigned int dec_status;
unsigned int enc_status;
dec_ctx = ndmp_door_decode_start(ptr, size);
if (dec_ctx == 0)
return;
req_type = ndmp_door_get_uint32(dec_ctx);
buflen = NDMP_DOOR_SIZE;
if ((buf = alloca(buflen)) == NULL) {
NDMP_LOG(LOG_DEBUG, "Out of memory.");
(void) ndmp_door_decode_finish(dec_ctx);
return;
}
enc_ctx = ndmp_door_encode_start(buf, buflen);
if (enc_ctx == 0) {
(void) ndmp_door_decode_finish(dec_ctx);
return;
}
if (req_type != NDMP_GET_STAT)
NDMP_LOG(LOG_DEBUG, "ndmp_door_server: req_type=%d", req_type);
switch (req_type) {
case NDMP_GET_DOOR_STATUS: {
ndmp_door_put_int32(enc_ctx, NDMP_DOOR_SRV_SUCCESS);
break;
}
case NDMP_DEVICES_GET_INFO: {
ndmp_door_put_int32(enc_ctx, NDMP_DOOR_SRV_SUCCESS);
ndmpd_get_devs(enc_ctx);
break;
}
case NDMP_SHOW: {
ndmp_door_put_int32(enc_ctx, NDMP_DOOR_SRV_SUCCESS);
ndmp_connect_list_get(enc_ctx);
break;
}
case NDMP_TERMINATE_SESSION_ID: {
int status, id;
id = ndmp_door_get_int32(dec_ctx);
status = ndmpd_connect_kill_id(id);
if (status == -1) /* session not found */
ndmp_door_put_int32(enc_ctx,
NDMP_DOOR_SRV_SUCCESS);
else
ndmp_door_put_int32(enc_ctx,
NDMP_DOOR_SRV_SUCCESS);
ndmp_door_put_int32(enc_ctx, status);
break;
}
case NDMP_GET_STAT:
ndmp_door_put_int32(enc_ctx, NDMP_DOOR_SRV_SUCCESS);
ndmp_door_put_uint32(enc_ctx, ndstat.ns_trun);
ndmp_door_put_uint32(enc_ctx, ndstat.ns_twait);
ndmp_door_put_uint32(enc_ctx, ndstat.ns_nbk);
ndmp_door_put_uint32(enc_ctx, ndstat.ns_nrs);
ndmp_door_put_uint32(enc_ctx, ndstat.ns_rfile);
ndmp_door_put_uint32(enc_ctx, ndstat.ns_wfile);
ndmp_door_put_uint64(enc_ctx, ndstat.ns_rdisk);
ndmp_door_put_uint64(enc_ctx, ndstat.ns_wdisk);
ndmp_door_put_uint64(enc_ctx, ndstat.ns_rtape);
ndmp_door_put_uint64(enc_ctx, ndstat.ns_wtape);
break;
default:
NDMP_LOG(LOG_DEBUG,
"ndmp_door_server: Invalid request type 0x%x", req_type);
goto decode_error;
}
if ((dec_status = ndmp_door_decode_finish(dec_ctx)) != 0)
goto decode_error;
if ((enc_status = ndmp_door_encode_finish(enc_ctx, &used)) != 0)
goto encode_error;
(void) door_return(buf, used, NULL, 0);
return;
decode_error:
ndmp_door_put_int32(enc_ctx, NDMP_DOOR_SRV_ERROR);
ndmp_door_put_uint32(enc_ctx, dec_status);
(void) ndmp_door_encode_finish(enc_ctx, &used);
(void) door_return(buf, used, NULL, 0);
return;
encode_error:
enc_ctx = ndmp_door_encode_start(buf, buflen);
ndmp_door_put_int32(enc_ctx, NDMP_DOOR_SRV_ERROR);
ndmp_door_put_uint32(enc_ctx, enc_status);
(void) ndmp_door_encode_finish(enc_ctx, &used);
(void) door_return(buf, used, NULL, 0);
}