12c1b14e5Sjose borrego /*
22c1b14e5Sjose borrego  * CDDL HEADER START
32c1b14e5Sjose borrego  *
42c1b14e5Sjose borrego  * The contents of this file are subject to the terms of the
52c1b14e5Sjose borrego  * Common Development and Distribution License (the "License").
62c1b14e5Sjose borrego  * You may not use this file except in compliance with the License.
72c1b14e5Sjose borrego  *
82c1b14e5Sjose borrego  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
92c1b14e5Sjose borrego  * or http://www.opensolaris.org/os/licensing.
102c1b14e5Sjose borrego  * See the License for the specific language governing permissions
112c1b14e5Sjose borrego  * and limitations under the License.
122c1b14e5Sjose borrego  *
132c1b14e5Sjose borrego  * When distributing Covered Code, include this CDDL HEADER in each
142c1b14e5Sjose borrego  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
152c1b14e5Sjose borrego  * If applicable, add the following below this CDDL HEADER, with the
162c1b14e5Sjose borrego  * fields enclosed by brackets "[]" replaced with your own identifying
172c1b14e5Sjose borrego  * information: Portions Copyright [yyyy] [name of copyright owner]
182c1b14e5Sjose borrego  *
192c1b14e5Sjose borrego  * CDDL HEADER END
202c1b14e5Sjose borrego  */
21148c5f43SAlan Wright 
222c1b14e5Sjose borrego /*
23148c5f43SAlan Wright  * Copyright (c) 2008, 2010, Oracle and/or its affiliates. All rights reserved.
24*3299f39fSGordon Ross  * Copyright 2013 Nexenta Systems, Inc.  All rights reserved.
252c1b14e5Sjose borrego  */
262c1b14e5Sjose borrego 
272c1b14e5Sjose borrego /*
282c1b14e5Sjose borrego  * Active Directory Setup RPC interface used by Windows 2000.
292c1b14e5Sjose borrego  */
302c1b14e5Sjose borrego 
312c1b14e5Sjose borrego #include <synch.h>
322c1b14e5Sjose borrego #include <strings.h>
332c1b14e5Sjose borrego #include <stdlib.h>
342c1b14e5Sjose borrego #include <netdb.h>
352c1b14e5Sjose borrego 
36*3299f39fSGordon Ross #include <libmlrpc/libmlrpc.h>
372c1b14e5Sjose borrego #include <smbsrv/libsmb.h>
382c1b14e5Sjose borrego #include <smbsrv/libmlsvc.h>
392c1b14e5Sjose borrego #include <smbsrv/ndl/dssetup.ndl>
402c1b14e5Sjose borrego #include <smbsrv/smbinfo.h>
412c1b14e5Sjose borrego #include <smbsrv/nmpipes.h>
422c1b14e5Sjose borrego 
432c1b14e5Sjose borrego int dssetup_get_domain_info(ds_primary_domain_info_t *);
442c1b14e5Sjose borrego 
452c1b14e5Sjose borrego static int dssetup_DsRoleGetPrimaryDomainInfo(void *, ndr_xa_t *);
462c1b14e5Sjose borrego static uint32_t dssetup_member_server(ds_primary_domain_info_t *, ndr_xa_t *);
472c1b14e5Sjose borrego static uint32_t dssetup_standalone_server(ds_primary_domain_info_t *,
482c1b14e5Sjose borrego     ndr_xa_t *);
492c1b14e5Sjose borrego 
508d7e4166Sjose borrego static ndr_stub_table_t dssetup_stub_table[] = {
512c1b14e5Sjose borrego 	{ dssetup_DsRoleGetPrimaryDomainInfo,
522c1b14e5Sjose borrego 	    DSSETUP_OPNUM_DsRoleGetPrimaryDomainInfo },
532c1b14e5Sjose borrego 	{0}
542c1b14e5Sjose borrego };
552c1b14e5Sjose borrego 
568d7e4166Sjose borrego static ndr_service_t dssetup_service = {
572c1b14e5Sjose borrego 	"DSSETUP",			/* name */
582c1b14e5Sjose borrego 	"Active Directory Setup",	/* desc */
592c1b14e5Sjose borrego 	"\\lsarpc",			/* endpoint */
602c1b14e5Sjose borrego 	PIPE_LSASS,			/* sec_addr_port */
618d7e4166Sjose borrego 	"3919286a-b10c-11d0-9ba8-00c04fd92ef5",	0,	/* abstract */
628d7e4166Sjose borrego 	NDR_TRANSFER_SYNTAX_UUID,		2,	/* transfer */
632c1b14e5Sjose borrego 	0,				/* no bind_instance_size */
642c1b14e5Sjose borrego 	0,				/* no bind_req() */
652c1b14e5Sjose borrego 	0,				/* no unbind_and_close() */
662c1b14e5Sjose borrego 	0,				/* use generic_call_stub() */
672c1b14e5Sjose borrego 	&TYPEINFO(dssetup_interface),	/* interface ti */
682c1b14e5Sjose borrego 	dssetup_stub_table		/* stub_table */
692c1b14e5Sjose borrego };
702c1b14e5Sjose borrego 
712c1b14e5Sjose borrego static ds_primary_domain_info_t ds_info;
722c1b14e5Sjose borrego static mutex_t ds_info_mtx;
732c1b14e5Sjose borrego 
742c1b14e5Sjose borrego /*
752c1b14e5Sjose borrego  * dssetup_initialize
762c1b14e5Sjose borrego  *
772c1b14e5Sjose borrego  * This function registers the DSSETUP interface with the RPC runtime
782c1b14e5Sjose borrego  * library. It must be called in order to use either the client side
792c1b14e5Sjose borrego  * or the server side functions.
802c1b14e5Sjose borrego  */
812c1b14e5Sjose borrego void
dssetup_initialize(void)822c1b14e5Sjose borrego dssetup_initialize(void)
832c1b14e5Sjose borrego {
842c1b14e5Sjose borrego 	dssetup_clear_domain_info();
858d7e4166Sjose borrego 	(void) ndr_svc_register(&dssetup_service);
862c1b14e5Sjose borrego }
872c1b14e5Sjose borrego 
882c1b14e5Sjose borrego void
dssetup_clear_domain_info(void)892c1b14e5Sjose borrego dssetup_clear_domain_info(void)
902c1b14e5Sjose borrego {
912c1b14e5Sjose borrego 	(void) mutex_lock(&ds_info_mtx);
922c1b14e5Sjose borrego 
932c1b14e5Sjose borrego 	free(ds_info.nt_domain);
942c1b14e5Sjose borrego 	free(ds_info.dns_domain);
952c1b14e5Sjose borrego 	free(ds_info.forest);
962c1b14e5Sjose borrego 	bzero(&ds_info, sizeof (ds_primary_domain_info_t));
972c1b14e5Sjose borrego 
982c1b14e5Sjose borrego 	(void) mutex_unlock(&ds_info_mtx);
992c1b14e5Sjose borrego }
1002c1b14e5Sjose borrego 
1012c1b14e5Sjose borrego /*
1022c1b14e5Sjose borrego  * Request for machine role and primary domain information.
1032c1b14e5Sjose borrego  */
1042c1b14e5Sjose borrego static int
dssetup_DsRoleGetPrimaryDomainInfo(void * arg,ndr_xa_t * mxa)1052c1b14e5Sjose borrego dssetup_DsRoleGetPrimaryDomainInfo(void *arg, ndr_xa_t *mxa)
1062c1b14e5Sjose borrego {
1072c1b14e5Sjose borrego 	dssetup_DsRoleGetPrimaryDomainInfo_t *param = arg;
1082c1b14e5Sjose borrego 	dssetup_GetPrimaryDomainInfo_t *info;
1092c1b14e5Sjose borrego 	ds_primary_domain_info_t *info1;
1102c1b14e5Sjose borrego 	uint32_t status;
1112c1b14e5Sjose borrego 	int security_mode;
1122c1b14e5Sjose borrego 
1138d7e4166Sjose borrego 	info = NDR_MALLOC(mxa, sizeof (dssetup_GetPrimaryDomainInfo_t));
1142c1b14e5Sjose borrego 	if (info == NULL) {
1152c1b14e5Sjose borrego 		status = NT_STATUS_NO_MEMORY;
1162c1b14e5Sjose borrego 	} else if (param->level != DS_ROLE_BASIC_INFORMATION) {
1172c1b14e5Sjose borrego 		status = NT_STATUS_INVALID_LEVEL;
1182c1b14e5Sjose borrego 	} else {
1192c1b14e5Sjose borrego 		info->switch_value = param->level;
1202c1b14e5Sjose borrego 		info1 = &info->ru.info1;
1212c1b14e5Sjose borrego 
1222c1b14e5Sjose borrego 		security_mode = smb_config_get_secmode();
1232c1b14e5Sjose borrego 
1242c1b14e5Sjose borrego 		if (security_mode == SMB_SECMODE_DOMAIN)
1252c1b14e5Sjose borrego 			status = dssetup_member_server(info1, mxa);
1262c1b14e5Sjose borrego 		else
1272c1b14e5Sjose borrego 			status = dssetup_standalone_server(info1, mxa);
1282c1b14e5Sjose borrego 	}
1292c1b14e5Sjose borrego 
1302c1b14e5Sjose borrego 	if (status != NT_STATUS_SUCCESS) {
1312c1b14e5Sjose borrego 		bzero(param, sizeof (dssetup_DsRoleGetPrimaryDomainInfo_t));
1322c1b14e5Sjose borrego 		param->status = NT_SC_ERROR(status);
1332c1b14e5Sjose borrego 	} else {
1342c1b14e5Sjose borrego 		param->info = info;
1352c1b14e5Sjose borrego 		param->status = NT_STATUS_SUCCESS;
1362c1b14e5Sjose borrego 	}
1372c1b14e5Sjose borrego 
1388d7e4166Sjose borrego 	return (NDR_DRC_OK);
1392c1b14e5Sjose borrego }
1402c1b14e5Sjose borrego 
1412c1b14e5Sjose borrego /*
1422c1b14e5Sjose borrego  * When the machine role is domain member:
1432c1b14e5Sjose borrego  * 	nt_domain must contain the NetBIOS domain name
1442c1b14e5Sjose borrego  * 	dns_domain must contain the DNS domain name (cannot be NULL)
1452c1b14e5Sjose borrego  * 	forest must contain the forest name (cannot be NULL)
1462c1b14e5Sjose borrego  *
1472c1b14e5Sjose borrego  * If DS_ROLE_PRIMARY_DOMAIN_GUID_PRESENT is set in flags, the domain_guid
1482c1b14e5Sjose borrego  * must contain the domain UUID.  Otherwise domain_guid is ignored.
1492c1b14e5Sjose borrego  */
1502c1b14e5Sjose borrego static uint32_t
dssetup_member_server(ds_primary_domain_info_t * info,ndr_xa_t * mxa)1512c1b14e5Sjose borrego dssetup_member_server(ds_primary_domain_info_t *info, ndr_xa_t *mxa)
1522c1b14e5Sjose borrego {
1532c1b14e5Sjose borrego 	char dns_domain[MAXHOSTNAMELEN];
1542c1b14e5Sjose borrego 	char nt_domain[MAXHOSTNAMELEN];
1552c1b14e5Sjose borrego 
1562c1b14e5Sjose borrego 	(void) mutex_lock(&ds_info_mtx);
1572c1b14e5Sjose borrego 
1582c1b14e5Sjose borrego 	if ((ds_info.flags & DS_ROLE_PRIMARY_DOMAIN_GUID_PRESENT) == 0) {
1592c1b14e5Sjose borrego 		/*
1602c1b14e5Sjose borrego 		 * If we don't have the domain GUID, try to get it from a
1612c1b14e5Sjose borrego 		 * domain controller. Otherwise, use local configuration.
1622c1b14e5Sjose borrego 		 */
1632c1b14e5Sjose borrego 		free(ds_info.nt_domain);
1642c1b14e5Sjose borrego 		free(ds_info.dns_domain);
1652c1b14e5Sjose borrego 		free(ds_info.forest);
1662c1b14e5Sjose borrego 		(void) dssetup_get_domain_info(&ds_info);
1672c1b14e5Sjose borrego 	}
1682c1b14e5Sjose borrego 
1692c1b14e5Sjose borrego 	if (ds_info.flags & DS_ROLE_PRIMARY_DOMAIN_GUID_PRESENT) {
1702c1b14e5Sjose borrego 		info->flags = DS_ROLE_PRIMARY_DOMAIN_GUID_PRESENT;
1718d7e4166Sjose borrego 		info->nt_domain = NDR_STRDUP(mxa, (char *)ds_info.nt_domain);
1728d7e4166Sjose borrego 		info->dns_domain = NDR_STRDUP(mxa, (char *)ds_info.dns_domain);
1738d7e4166Sjose borrego 		info->forest = NDR_STRDUP(mxa, (char *)ds_info.forest);
1742c1b14e5Sjose borrego 		bcopy(&ds_info.domain_guid, &info->domain_guid,
1752c1b14e5Sjose borrego 		    sizeof (ndr_uuid_t));
1762c1b14e5Sjose borrego 	} else {
1772c1b14e5Sjose borrego 		if (smb_getdomainname(nt_domain, MAXHOSTNAMELEN) != 0) {
1782c1b14e5Sjose borrego 			(void) mutex_unlock(&ds_info_mtx);
1792c1b14e5Sjose borrego 			return (NT_STATUS_CANT_ACCESS_DOMAIN_INFO);
1802c1b14e5Sjose borrego 		}
1812c1b14e5Sjose borrego 
1822c1b14e5Sjose borrego 		if (smb_getfqdomainname(dns_domain, MAXHOSTNAMELEN) != 0) {
1832c1b14e5Sjose borrego 			(void) mutex_unlock(&ds_info_mtx);
1842c1b14e5Sjose borrego 			return (NT_STATUS_CANT_ACCESS_DOMAIN_INFO);
1852c1b14e5Sjose borrego 		}
1862c1b14e5Sjose borrego 
187bbf6f00cSJordan Brown 		(void) smb_strlwr(dns_domain);
1882c1b14e5Sjose borrego 
1892c1b14e5Sjose borrego 		info->flags = 0;
1908d7e4166Sjose borrego 		info->nt_domain = NDR_STRDUP(mxa, nt_domain);
1918d7e4166Sjose borrego 		info->dns_domain = NDR_STRDUP(mxa, dns_domain);
1928d7e4166Sjose borrego 		info->forest = NDR_STRDUP(mxa, dns_domain);
1932c1b14e5Sjose borrego 		bzero(&info->domain_guid, sizeof (ndr_uuid_t));
1942c1b14e5Sjose borrego 	}
1952c1b14e5Sjose borrego 
1962c1b14e5Sjose borrego 	(void) mutex_unlock(&ds_info_mtx);
1972c1b14e5Sjose borrego 
1982c1b14e5Sjose borrego 	if (info->nt_domain == NULL ||
1992c1b14e5Sjose borrego 	    info->dns_domain == NULL ||
2002c1b14e5Sjose borrego 	    info->forest == NULL)
2012c1b14e5Sjose borrego 		return (NT_STATUS_NO_MEMORY);
2022c1b14e5Sjose borrego 
2032c1b14e5Sjose borrego 	info->role = DS_ROLE_MEMBER_SERVER;
2042c1b14e5Sjose borrego 	return (NT_STATUS_SUCCESS);
2052c1b14e5Sjose borrego }
2062c1b14e5Sjose borrego 
2072c1b14e5Sjose borrego /*
2082c1b14e5Sjose borrego  * When the machine role is standalone:
2092c1b14e5Sjose borrego  * 	nt_domain must contain the NetBIOS workgroup name
2102c1b14e5Sjose borrego  * 	dns_domain must be NULL
2112c1b14e5Sjose borrego  * 	forest must be NULL
2122c1b14e5Sjose borrego  *
2132c1b14e5Sjose borrego  * We don't maintain a domain GUID.  When DS_ROLE_PRIMARY_DOMAIN_GUID_PRESENT
2142c1b14e5Sjose borrego  * is not set in flags, domain_guid is ignored.
2152c1b14e5Sjose borrego  */
2162c1b14e5Sjose borrego static uint32_t
dssetup_standalone_server(ds_primary_domain_info_t * info,ndr_xa_t * mxa)2172c1b14e5Sjose borrego dssetup_standalone_server(ds_primary_domain_info_t *info, ndr_xa_t *mxa)
2182c1b14e5Sjose borrego {
2192c1b14e5Sjose borrego 	char nt_domain[MAXHOSTNAMELEN];
2202c1b14e5Sjose borrego 
2212c1b14e5Sjose borrego 	if (smb_getdomainname(nt_domain, MAXHOSTNAMELEN) != 0)
2222c1b14e5Sjose borrego 		return (NT_STATUS_CANT_ACCESS_DOMAIN_INFO);
2232c1b14e5Sjose borrego 
2248d7e4166Sjose borrego 	info->nt_domain = NDR_STRDUP(mxa, nt_domain);
2252c1b14e5Sjose borrego 	if (info->nt_domain == NULL)
2262c1b14e5Sjose borrego 		return (NT_STATUS_NO_MEMORY);
2272c1b14e5Sjose borrego 
2282c1b14e5Sjose borrego 	info->role = DS_ROLE_STANDALONE_SERVER;
2292c1b14e5Sjose borrego 	info->flags = 0;
2302c1b14e5Sjose borrego 	info->dns_domain = NULL;
2312c1b14e5Sjose borrego 	info->forest = NULL;
2322c1b14e5Sjose borrego 	bzero(&info->domain_guid, sizeof (ndr_uuid_t));
2332c1b14e5Sjose borrego 	return (NT_STATUS_SUCCESS);
2342c1b14e5Sjose borrego }
235