#include <assert.h>
#include <syslog.h>
#include <door.h>
#include <fcntl.h>
#include <string.h>
#include <strings.h>
#include <stdlib.h>
#include <unistd.h>
#include <errno.h>
#include <sys/mman.h>
#include <smb/wintypes.h>
#include <smbsrv/libsmb.h>
#include <smbsrv/smb_door.h>
static int smb_door_call(uint32_t, void *, xdrproc_t, void *, xdrproc_t);
static int smb_door_call_private(int, smb_doorarg_t *);
static int smb_door_encode(smb_doorarg_t *, uint32_t);
static int smb_door_decode(smb_doorarg_t *);
static void smb_door_sethdr(smb_doorhdr_t *, uint32_t, uint32_t);
static boolean_t smb_door_chkhdr(smb_doorarg_t *, smb_doorhdr_t *);
static void smb_door_free(door_arg_t *arg);
static int smb_lookup_name_int(const char *name, sid_type_t sidtype,
lsa_account_t *acct, int);
static int smb_lookup_sid_int(const char *sid, lsa_account_t *acct, int);
int
smb_lookup_sid(const char *sid, lsa_account_t *acct)
{
return (smb_lookup_sid_int(sid, acct, SMB_DR_LOOKUP_SID));
}
int
smb_lookup_lsid(const char *sid, lsa_account_t *acct)
{
return (smb_lookup_sid_int(sid, acct, SMB_DR_LOOKUP_LSID));
}
static int
smb_lookup_sid_int(const char *sid, lsa_account_t *acct, int dop)
{
int rc;
assert((sid != NULL) && (acct != NULL));
bzero(acct, sizeof (lsa_account_t));
(void) strlcpy(acct->a_sid, sid, SMB_SID_STRSZ);
rc = smb_door_call(dop, acct, lsa_account_xdr,
acct, lsa_account_xdr);
if (rc != 0)
syslog(LOG_DEBUG, "smb_lookup_sid: %m");
return (rc);
}
int
smb_lookup_name(const char *name, sid_type_t sidtype, lsa_account_t *acct)
{
return (smb_lookup_name_int(name, sidtype, acct, SMB_DR_LOOKUP_NAME));
}
int
smb_lookup_lname(const char *name, sid_type_t sidtype, lsa_account_t *acct)
{
return (smb_lookup_name_int(name, sidtype, acct, SMB_DR_LOOKUP_LNAME));
}
static int
smb_lookup_name_int(const char *name, sid_type_t sidtype, lsa_account_t *acct,
int dop)
{
char tmp[MAXNAMELEN];
char *dp = NULL;
char *np = NULL;
int rc;
assert((name != NULL) && (acct != NULL));
(void) strlcpy(tmp, name, MAXNAMELEN);
smb_name_parse(tmp, &np, &dp);
bzero(acct, sizeof (lsa_account_t));
acct->a_sidtype = sidtype;
if (dp != NULL && np != NULL) {
(void) strlcpy(acct->a_domain, dp, MAXNAMELEN);
(void) strlcpy(acct->a_name, np, MAXNAMELEN);
} else {
(void) strlcpy(acct->a_name, name, MAXNAMELEN);
}
rc = smb_door_call(dop, acct, lsa_account_xdr,
acct, lsa_account_xdr);
if (rc != 0)
syslog(LOG_DEBUG, "smb_lookup_name: %m");
return (rc);
}
int
smb_join(smb_joininfo_t *jdi, smb_joinres_t *jres)
{
int rc;
rc = smb_door_call(SMB_DR_JOIN, jdi, smb_joininfo_xdr,
jres, smb_joinres_xdr);
if (rc != 0) {
syslog(LOG_DEBUG, "smb_join: %m");
jres->status = NT_STATUS_SERVER_DISABLED;
return (rc);
}
return (0);
}
uint32_t
smb_get_dcinfo(char *namebuf, uint32_t namebuflen, smb_inaddr_t *ipaddr)
{
smb_string_t dcname;
struct hostent *h;
int rc;
assert((namebuf != NULL) && (namebuflen != 0));
*namebuf = '\0';
bzero(&dcname, sizeof (smb_string_t));
rc = smb_door_call(SMB_DR_GET_DCINFO, NULL, NULL,
&dcname, smb_string_xdr);
if (rc != 0) {
syslog(LOG_DEBUG, "smb_get_dcinfo: %m");
if (dcname.buf)
xdr_free(smb_string_xdr, (char *)&dcname);
return (NT_STATUS_INTERNAL_ERROR);
}
if (dcname.buf) {
(void) strlcpy(namebuf, dcname.buf, namebuflen);
if ((h = smb_gethostbyname(dcname.buf, &rc)) == NULL) {
bzero(ipaddr, sizeof (smb_inaddr_t));
} else {
(void) memcpy(ipaddr, h->h_addr, h->h_length);
ipaddr->a_family = h->h_addrtype;
freehostent(h);
}
xdr_free(smb_string_xdr, (char *)&dcname);
}
return (NT_STATUS_SUCCESS);
}
bool_t
smb_joininfo_xdr(XDR *xdrs, smb_joininfo_t *objp)
{
if (!xdr_uint32_t(xdrs, &objp->mode))
return (FALSE);
if (!xdr_vector(xdrs, (char *)objp->domain_name, MAXHOSTNAMELEN,
sizeof (char), (xdrproc_t)xdr_char))
return (FALSE);
if (!xdr_vector(xdrs, (char *)objp->container_name, MAXHOSTNAMELEN,
sizeof (char), (xdrproc_t)xdr_char))
return (FALSE);
if (!xdr_vector(xdrs, (char *)objp->domain_username,
SMB_USERNAME_MAXLEN + 1, sizeof (char), (xdrproc_t)xdr_char))
return (FALSE);
if (!xdr_vector(xdrs, (char *)objp->domain_passwd,
SMB_PASSWD_MAXLEN + 1, sizeof (char), (xdrproc_t)xdr_char))
return (FALSE);
return (TRUE);
}
bool_t
smb_joinres_xdr(XDR *xdrs, smb_joinres_t *objp)
{
if (!xdr_uint32_t(xdrs, &objp->status))
return (FALSE);
if (!xdr_int(xdrs, &objp->join_err))
return (FALSE);
if (!xdr_vector(xdrs, (char *)objp->dc_name, MAXHOSTNAMELEN,
sizeof (char), (xdrproc_t)xdr_char))
return (FALSE);
return (TRUE);
}
boolean_t
smb_find_ads_server(char *fqdn, char *buf, int buflen)
{
smb_string_t server;
smb_string_t domain;
boolean_t found = B_FALSE;
int rc;
if (fqdn == NULL || buf == NULL) {
if (buf)
*buf = '\0';
return (B_FALSE);
}
bzero(&server, sizeof (smb_string_t));
*buf = '\0';
domain.buf = fqdn;
rc = smb_door_call(SMB_DR_ADS_FIND_HOST, &domain, smb_string_xdr,
&server, smb_string_xdr);
if (rc != 0)
syslog(LOG_DEBUG, "smb_find_ads_server: %m");
if (server.buf != NULL) {
if (*server.buf != '\0') {
(void) strlcpy(buf, server.buf, buflen);
found = B_TRUE;
}
xdr_free(smb_string_xdr, (char *)&server);
}
return (found);
}
int
smb_notify_dc_changed(void)
{
int rc;
rc = smb_door_call(SMB_DR_NOTIFY_DC_CHANGED,
NULL, NULL, NULL, NULL);
if (rc != 0) {
rc = errno;
if (rc == 0)
rc = EPERM;
syslog(LOG_DEBUG, "smb_notify_dc_changed: rc=%d", rc);
return (rc);
}
return (0);
}
static int
smb_door_call(uint32_t cmd, void *req_data, xdrproc_t req_xdr,
void *rsp_data, xdrproc_t rsp_xdr)
{
smb_doorarg_t da;
int fd;
int rc;
char *door_name;
bzero(&da, sizeof (smb_doorarg_t));
da.da_opcode = cmd;
da.da_opname = smb_doorhdr_opname(cmd);
da.da_req_xdr = req_xdr;
da.da_rsp_xdr = rsp_xdr;
da.da_req_data = req_data;
da.da_rsp_data = rsp_data;
if ((req_data == NULL && req_xdr != NULL) ||
(rsp_data == NULL && rsp_xdr != NULL)) {
errno = EINVAL;
syslog(LOG_DEBUG, "smb_door_call[%s]: %m", da.da_opname);
return (-1);
}
door_name = getenv("SMBD_DOOR_NAME");
if (door_name == NULL)
door_name = SMBD_DOOR_NAME;
if ((fd = open(door_name, O_RDONLY)) < 0) {
syslog(LOG_DEBUG, "smb_door_call[%s]: %m", da.da_opname);
return (-1);
}
if (smb_door_encode(&da, cmd) != 0) {
syslog(LOG_DEBUG, "smb_door_call[%s]: %m", da.da_opname);
(void) close(fd);
return (-1);
}
if (smb_door_call_private(fd, &da) != 0) {
syslog(LOG_DEBUG, "smb_door_call[%s]: %m", da.da_opname);
smb_door_free(&da.da_arg);
(void) close(fd);
return (-1);
}
if ((rc = smb_door_decode(&da)) != 0)
syslog(LOG_DEBUG, "smb_door_call[%s]: %m", da.da_opname);
smb_door_free(&da.da_arg);
(void) close(fd);
return (rc);
}
static int
smb_door_call_private(int fd, smb_doorarg_t *da)
{
door_arg_t door_arg;
int rc;
int i;
bcopy(&da->da_arg, &door_arg, sizeof (door_arg_t));
for (i = 0; i < SMB_DOOR_CALL_RETRIES; ++i) {
errno = 0;
if ((rc = door_call(fd, &door_arg)) == 0)
break;
if (errno != EAGAIN && errno != EINTR)
return (-1);
}
if (rc != 0 || door_arg.data_size == 0 || door_arg.rsize == 0) {
if (errno == 0)
errno = EIO;
return (-1);
}
da->da_arg.rbuf = door_arg.data_ptr;
da->da_arg.rsize = door_arg.rsize;
return (rc);
}
static int
smb_door_encode(smb_doorarg_t *da, uint32_t cmd)
{
XDR xdrs;
char *buf;
uint32_t buflen;
buflen = xdr_sizeof(smb_doorhdr_xdr, &da->da_hdr);
if (da->da_req_xdr != NULL)
buflen += xdr_sizeof(da->da_req_xdr, da->da_req_data);
smb_door_sethdr(&da->da_hdr, cmd, buflen);
if ((buf = malloc(buflen)) == NULL)
return (-1);
xdrmem_create(&xdrs, buf, buflen, XDR_ENCODE);
if (!smb_doorhdr_xdr(&xdrs, &da->da_hdr)) {
errno = EPROTO;
free(buf);
xdr_destroy(&xdrs);
return (-1);
}
if (da->da_req_xdr != NULL) {
if (!da->da_req_xdr(&xdrs, da->da_req_data)) {
errno = EPROTO;
free(buf);
xdr_destroy(&xdrs);
return (-1);
}
}
da->da_arg.data_ptr = buf;
da->da_arg.data_size = buflen;
da->da_arg.desc_ptr = NULL;
da->da_arg.desc_num = 0;
da->da_arg.rbuf = buf;
da->da_arg.rsize = buflen;
xdr_destroy(&xdrs);
return (0);
}
static int
smb_door_decode(smb_doorarg_t *da)
{
XDR xdrs;
smb_doorhdr_t hdr;
char *rbuf = da->da_arg.rbuf;
uint32_t rsize = da->da_arg.rsize;
if (rbuf == NULL || rsize == 0) {
errno = EINVAL;
return (-1);
}
xdrmem_create(&xdrs, rbuf, rsize, XDR_DECODE);
if (!smb_doorhdr_xdr(&xdrs, &hdr)) {
errno = EPROTO;
xdr_destroy(&xdrs);
return (-1);
}
if (!smb_door_chkhdr(da, &hdr)) {
errno = EPROTO;
xdr_destroy(&xdrs);
return (-1);
}
if (da->da_rsp_xdr != NULL) {
if (!da->da_rsp_xdr(&xdrs, da->da_rsp_data)) {
errno = EPROTO;
xdr_destroy(&xdrs);
return (-1);
}
}
xdr_destroy(&xdrs);
return (0);
}
static void
smb_door_sethdr(smb_doorhdr_t *hdr, uint32_t cmd, uint32_t datalen)
{
bzero(hdr, sizeof (smb_doorhdr_t));
hdr->dh_magic = SMB_DOOR_HDR_MAGIC;
hdr->dh_flags = SMB_DF_USERSPACE;
hdr->dh_op = cmd;
hdr->dh_txid = smb_get_txid();
hdr->dh_datalen = datalen;
hdr->dh_door_rc = SMB_DOP_NOT_CALLED;
}
static boolean_t
smb_door_chkhdr(smb_doorarg_t *da, smb_doorhdr_t *hdr)
{
if ((hdr->dh_magic != SMB_DOOR_HDR_MAGIC) ||
(hdr->dh_op != da->da_hdr.dh_op) ||
(hdr->dh_txid != da->da_hdr.dh_txid)) {
syslog(LOG_DEBUG, "smb_door_chkhdr[%s]: invalid header",
da->da_opname);
return (B_FALSE);
}
if (hdr->dh_door_rc != SMB_DOP_SUCCESS) {
syslog(LOG_DEBUG, "smb_door_chkhdr[%s]: call status=%d",
da->da_opname, hdr->dh_door_rc);
return (B_FALSE);
}
return (B_TRUE);
}
static void
smb_door_free(door_arg_t *arg)
{
if (arg->rbuf && (arg->rbuf != arg->data_ptr))
(void) munmap(arg->rbuf, arg->rsize);
free(arg->data_ptr);
}