2005-09-22 18:35:08 +00:00
/*
Unix SMB / CIFS implementation .
Main winbindd samba3 server routines
Copyright ( C ) Stefan Metzmacher 2005
Copyright ( C ) Volker Lendecke 2005
This program is free software ; you can redistribute it and / or modify
it under the terms of the GNU General Public License as published by
2007-07-10 02:07:03 +00:00
the Free Software Foundation ; either version 3 of the License , or
2005-09-22 18:35:08 +00:00
( at your option ) any later version .
This program is distributed in the hope that it will be useful ,
but WITHOUT ANY WARRANTY ; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
GNU General Public License for more details .
You should have received a copy of the GNU General Public License
2007-07-10 02:07:03 +00:00
along with this program . If not , see < http : //www.gnu.org/licenses/>.
2005-09-22 18:35:08 +00:00
*/
# include "includes.h"
2007-06-02 11:38:27 +00:00
# include "nsswitch/winbind_nss_config.h"
2005-09-22 18:35:08 +00:00
# include "nsswitch/winbindd_nss.h"
# include "winbind/wb_server.h"
2006-01-12 09:38:35 +00:00
# include "smbd/service_stream.h"
# include "lib/stream/packet.h"
2005-09-22 18:35:08 +00:00
2006-01-12 09:38:35 +00:00
/*
2006-07-26 05:19:36 +00:00
work out if a packet is complete for protocols that use a 32 bit host byte
2006-01-12 09:38:35 +00:00
order length
*/
NTSTATUS wbsrv_samba3_packet_full_request ( void * private , DATA_BLOB blob , size_t * size )
2005-09-22 18:35:08 +00:00
{
2006-01-12 09:38:35 +00:00
uint32_t * len ;
if ( blob . length < 4 ) {
return STATUS_MORE_ENTRIES ;
}
len = ( uint32_t * ) blob . data ;
* size = ( * len ) ;
if ( * size > blob . length ) {
return STATUS_MORE_ENTRIES ;
}
return NT_STATUS_OK ;
2005-09-22 18:35:08 +00:00
}
2006-01-12 09:38:35 +00:00
NTSTATUS wbsrv_samba3_pull_request ( DATA_BLOB blob , struct wbsrv_connection * wbconn ,
struct wbsrv_samba3_call * * _call )
2005-09-22 18:35:08 +00:00
{
2006-01-12 09:38:35 +00:00
struct wbsrv_samba3_call * call ;
2005-09-22 18:35:08 +00:00
2006-01-12 09:38:35 +00:00
if ( blob . length ! = sizeof ( call - > request ) ) {
2005-11-30 03:20:25 +00:00
DEBUG ( 0 , ( " wbsrv_samba3_pull_request: invalid blob length %lu should be %lu \n "
2005-09-22 18:35:08 +00:00
" make sure you use the correct winbind client tools! \n " ,
2006-01-12 09:38:35 +00:00
( long ) blob . length , ( long ) sizeof ( call - > request ) ) ) ;
2005-09-22 18:35:08 +00:00
return NT_STATUS_INVALID_PARAMETER ;
}
2006-01-12 09:38:35 +00:00
call = talloc_zero ( wbconn , struct wbsrv_samba3_call ) ;
2005-09-22 18:35:08 +00:00
NT_STATUS_HAVE_NO_MEMORY ( call ) ;
/* the packet layout is the same as the in memory layout of the request, so just copy it */
2006-01-12 09:38:35 +00:00
memcpy ( & call - > request , blob . data , sizeof ( call - > request ) ) ;
2005-09-22 18:35:08 +00:00
2006-01-12 09:38:35 +00:00
call - > wbconn = wbconn ;
call - > event_ctx = call - > wbconn - > conn - > event . ctx ;
2005-09-22 18:35:08 +00:00
* _call = call ;
return NT_STATUS_OK ;
}
2006-01-12 09:38:35 +00:00
NTSTATUS wbsrv_samba3_handle_call ( struct wbsrv_samba3_call * s3call )
2005-09-22 18:35:08 +00:00
{
DEBUG ( 10 , ( " Got winbind samba3 request %d \n " , s3call - > request . cmd ) ) ;
2005-10-19 13:45:44 +00:00
s3call - > response . length = sizeof ( s3call - > response ) ;
2005-09-22 18:35:08 +00:00
switch ( s3call - > request . cmd ) {
case WINBINDD_INTERFACE_VERSION :
return wbsrv_samba3_interface_version ( s3call ) ;
2005-11-05 23:46:57 +00:00
#if 0
2005-09-26 13:42:42 +00:00
case WINBINDD_CHECK_MACHACC :
return wbsrv_samba3_check_machacc ( s3call ) ;
2005-11-05 23:46:57 +00:00
# endif
2005-09-22 18:35:08 +00:00
case WINBINDD_PING :
return wbsrv_samba3_ping ( s3call ) ;
2005-09-25 21:01:56 +00:00
2005-09-26 13:42:42 +00:00
case WINBINDD_INFO :
2005-10-14 21:41:08 +00:00
return wbsrv_samba3_info ( s3call ) ;
2005-09-26 13:42:42 +00:00
case WINBINDD_DOMAIN_NAME :
return wbsrv_samba3_domain_name ( s3call ) ;
case WINBINDD_NETBIOS_NAME :
return wbsrv_samba3_netbios_name ( s3call ) ;
case WINBINDD_PRIV_PIPE_DIR :
return wbsrv_samba3_priv_pipe_dir ( s3call ) ;
2005-10-03 17:36:49 +00:00
case WINBINDD_LOOKUPNAME :
return wbsrv_samba3_lookupname ( s3call ) ;
2005-10-09 12:13:05 +00:00
2005-10-19 13:45:44 +00:00
case WINBINDD_LOOKUPSID :
return wbsrv_samba3_lookupsid ( s3call ) ;
2005-10-09 12:13:05 +00:00
case WINBINDD_PAM_AUTH :
return wbsrv_samba3_pam_auth ( s3call ) ;
case WINBINDD_PAM_AUTH_CRAP :
return wbsrv_samba3_pam_auth_crap ( s3call ) ;
2005-10-15 22:01:15 +00:00
case WINBINDD_GETDCNAME :
return wbsrv_samba3_getdcname ( s3call ) ;
2005-10-16 12:43:09 +00:00
case WINBINDD_GETUSERDOMGROUPS :
return wbsrv_samba3_userdomgroups ( s3call ) ;
2005-10-19 13:45:44 +00:00
case WINBINDD_GETUSERSIDS :
return wbsrv_samba3_usersids ( s3call ) ;
2005-10-19 21:53:03 +00:00
case WINBINDD_LIST_TRUSTDOM :
return wbsrv_samba3_list_trustdom ( s3call ) ;
2005-11-10 03:48:56 +00:00
2007-08-16 10:40:04 +00:00
case WINBINDD_LIST_USERS :
return wbsrv_samba3_list_users ( s3call ) ;
2005-11-10 03:48:56 +00:00
case WINBINDD_GETPWNAM :
2007-06-29 11:07:19 +00:00
return wbsrv_samba3_getpwnam ( s3call ) ;
2005-11-10 03:48:56 +00:00
case WINBINDD_GETPWUID :
2007-06-29 11:07:19 +00:00
return wbsrv_samba3_getpwuid ( s3call ) ;
case WINBINDD_SETPWENT :
return wbsrv_samba3_setpwent ( s3call ) ;
case WINBINDD_GETPWENT :
return wbsrv_samba3_getpwent ( s3call ) ;
case WINBINDD_ENDPWENT :
return wbsrv_samba3_endpwent ( s3call ) ;
2005-11-10 03:48:56 +00:00
case WINBINDD_GETGRNAM :
2007-06-29 11:07:19 +00:00
return wbsrv_samba3_getgrnam ( s3call ) ;
2005-11-10 03:48:56 +00:00
case WINBINDD_GETGRGID :
2007-06-29 11:07:19 +00:00
return wbsrv_samba3_getgrgid ( s3call ) ;
2005-11-10 03:48:56 +00:00
case WINBINDD_GETGROUPS :
2007-06-29 11:07:19 +00:00
return wbsrv_samba3_getgroups ( s3call ) ;
2005-11-10 03:48:56 +00:00
case WINBINDD_SETGRENT :
2007-06-29 11:07:19 +00:00
return wbsrv_samba3_setgrent ( s3call ) ;
2005-11-10 03:48:56 +00:00
case WINBINDD_GETGRENT :
2007-06-29 11:07:19 +00:00
return wbsrv_samba3_getgrent ( s3call ) ;
case WINBINDD_ENDGRENT :
return wbsrv_samba3_endgrent ( s3call ) ;
2007-08-13 15:58:41 +00:00
case WINBINDD_SID_TO_UID :
case WINBINDD_DUAL_SID2UID :
return wbsrv_samba3_sid2uid ( s3call ) ;
case WINBINDD_SID_TO_GID :
case WINBINDD_DUAL_SID2GID :
return wbsrv_samba3_sid2gid ( s3call ) ;
2007-08-13 16:07:47 +00:00
case WINBINDD_UID_TO_SID :
case WINBINDD_DUAL_UID2SID :
return wbsrv_samba3_uid2sid ( s3call ) ;
2007-08-13 16:20:26 +00:00
case WINBINDD_GID_TO_SID :
case WINBINDD_DUAL_GID2SID :
return wbsrv_samba3_gid2sid ( s3call ) ;
2007-06-29 11:07:19 +00:00
/* Unimplemented commands */
2005-11-10 03:48:56 +00:00
case WINBINDD_PAM_CHAUTHTOK :
2007-06-02 11:38:27 +00:00
case WINBINDD_PAM_LOGOFF :
case WINBINDD_PAM_CHNG_PSWD_AUTH_CRAP :
2005-11-10 03:48:56 +00:00
case WINBINDD_LIST_GROUPS :
2007-06-02 11:38:27 +00:00
case WINBINDD_LOOKUPRIDS :
case WINBINDD_SIDS_TO_XIDS :
case WINBINDD_ALLOCATE_UID :
case WINBINDD_ALLOCATE_GID :
case WINBINDD_SET_MAPPING :
case WINBINDD_SET_HWM :
case WINBINDD_DUMP_MAPS :
2005-11-10 03:48:56 +00:00
case WINBINDD_CHECK_MACHACC :
case WINBINDD_DOMAIN_INFO :
case WINBINDD_SHOW_SEQUENCE :
case WINBINDD_WINS_BYIP :
case WINBINDD_WINS_BYNAME :
case WINBINDD_GETGRLST :
case WINBINDD_INIT_CONNECTION :
2007-06-02 11:38:27 +00:00
case WINBINDD_DUAL_SIDS2XIDS :
case WINBINDD_DUAL_SET_MAPPING :
case WINBINDD_DUAL_SET_HWM :
case WINBINDD_DUAL_DUMP_MAPS :
2005-11-10 03:48:56 +00:00
case WINBINDD_DUAL_UID2NAME :
case WINBINDD_DUAL_NAME2UID :
case WINBINDD_DUAL_GID2NAME :
case WINBINDD_DUAL_NAME2GID :
case WINBINDD_DUAL_USERINFO :
case WINBINDD_DUAL_GETSIDALIASES :
2007-06-02 11:38:27 +00:00
case WINBINDD_CCACHE_NTLMAUTH :
2005-11-10 03:48:56 +00:00
case WINBINDD_NUM_CMDS :
DEBUG ( 10 , ( " Unimplemented winbind samba3 request %d \n " ,
s3call - > request . cmd ) ) ;
break ;
2005-09-22 18:35:08 +00:00
}
s3call - > response . result = WINBINDD_ERROR ;
return NT_STATUS_OK ;
}
2006-01-12 09:38:35 +00:00
static NTSTATUS wbsrv_samba3_push_reply ( struct wbsrv_samba3_call * call , TALLOC_CTX * mem_ctx , DATA_BLOB * _blob )
2005-09-22 18:35:08 +00:00
{
DATA_BLOB blob ;
uint8_t * extra_data ;
size_t extra_data_len = 0 ;
2007-06-02 11:38:27 +00:00
extra_data = call - > response . extra_data . data ;
2005-09-22 18:35:08 +00:00
if ( extra_data ) {
2006-01-12 09:38:35 +00:00
extra_data_len = call - > response . length -
sizeof ( call - > response ) ;
2005-09-22 18:35:08 +00:00
}
2006-01-12 09:38:35 +00:00
blob = data_blob_talloc ( mem_ctx , NULL , call - > response . length ) ;
2005-09-22 18:35:08 +00:00
NT_STATUS_HAVE_NO_MEMORY ( blob . data ) ;
/* don't push real pointer values into sockets */
if ( extra_data ) {
2007-06-02 11:38:27 +00:00
call - > response . extra_data . data = ( void * ) 0xFFFFFFFF ;
2005-09-22 18:35:08 +00:00
}
2006-01-12 09:38:35 +00:00
memcpy ( blob . data , & call - > response , sizeof ( call - > response ) ) ;
2005-09-22 18:35:08 +00:00
/* set back the pointer */
2007-06-02 11:38:27 +00:00
call - > response . extra_data . data = extra_data ;
2005-09-22 18:35:08 +00:00
if ( extra_data ) {
2006-01-12 09:38:35 +00:00
memcpy ( blob . data + sizeof ( call - > response ) , extra_data , extra_data_len ) ;
2005-09-22 18:35:08 +00:00
}
* _blob = blob ;
return NT_STATUS_OK ;
}
2006-01-12 09:38:35 +00:00
/*
* queue a wbsrv_call reply on a wbsrv_connection
* NOTE : that this implies talloc_free ( call ) ,
* use talloc_reference ( call ) if you need it after
* calling wbsrv_queue_reply
*/
NTSTATUS wbsrv_samba3_send_reply ( struct wbsrv_samba3_call * call )
{
struct wbsrv_connection * wbconn = call - > wbconn ;
DATA_BLOB rep ;
NTSTATUS status ;
status = wbsrv_samba3_push_reply ( call , call , & rep ) ;
NT_STATUS_NOT_OK_RETURN ( status ) ;
status = packet_send ( call - > wbconn - > packet , rep ) ;
talloc_free ( call ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
wbsrv_terminate_connection ( wbconn ,
" failed to packet_send winbindd reply " ) ;
return status ;
}
/* the call isn't needed any more */
return status ;
}
NTSTATUS wbsrv_samba3_process ( void * private , DATA_BLOB blob )
{
NTSTATUS status ;
struct wbsrv_connection * wbconn = talloc_get_type ( private ,
struct wbsrv_connection ) ;
struct wbsrv_samba3_call * call ;
status = wbsrv_samba3_pull_request ( blob , wbconn , & call ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
status = wbsrv_samba3_handle_call ( call ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
talloc_free ( call ) ;
return status ;
}
if ( call - > flags & WBSRV_CALL_FLAGS_REPLY_ASYNC ) {
return NT_STATUS_OK ;
}
status = wbsrv_samba3_send_reply ( call ) ;
return status ;
}