/* * Unix SMB/CIFS implementation. * RPC Pipe client / server routines * Copyright (C) Andrew Tridgell 1992-1997, * Copyright (C) Luke Kenneth Casson Leighton 1996-1997, * Copyright (C) Paul Ashton 1997, * Copyright (C) Jeremy Allison 2001, * Copyright (C) Jim McDonough 2003. * Copyright (C) Gera;d (Jerry) Carter 2006. * * 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 * the Free Software Foundation; either version 3 of the License, or * (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 * along with this program; if not, see . */ /* This is the interface to the srvsvc pipe. */ #include "includes.h" #undef DBGC_CLASS #define DBGC_CLASS DBGC_RPC_SRV static bool proxy_srvsvc_call(pipes_struct *p, uint8 opnum) { struct api_struct *fns; int n_fns; srvsvc_get_pipe_fns(&fns, &n_fns); if (opnum >= n_fns) return False; if (fns[opnum].opnum != opnum) { smb_panic("SRVSVC function table not sorted\n"); } return fns[opnum].fn(p); } /******************************************************************* api_srv_net_srv_get_info ********************************************************************/ static bool api_srv_net_srv_get_info(pipes_struct *p) { return proxy_srvsvc_call(p, NDR_SRVSVC_NETSRVGETINFO); } /******************************************************************* api_srv_net_srv_get_info ********************************************************************/ static bool api_srv_net_srv_set_info(pipes_struct *p) { return proxy_srvsvc_call(p, NDR_SRVSVC_NETSRVSETINFO); } /******************************************************************* api_srv_net_file_enum ********************************************************************/ static bool api_srv_net_file_enum(pipes_struct *p) { SRV_Q_NET_FILE_ENUM q_u; SRV_R_NET_FILE_ENUM r_u; prs_struct *data = &p->in_data.data; prs_struct *rdata = &p->out_data.rdata; ZERO_STRUCT(q_u); ZERO_STRUCT(r_u); /* grab the net file enum */ if (!srv_io_q_net_file_enum("", &q_u, data, 0)) return False; r_u.status = _srv_net_file_enum(p, &q_u, &r_u); /* store the response in the SMB stream */ if(!srv_io_r_net_file_enum("", &r_u, rdata, 0)) return False; return True; } /******************************************************************* api_srv_net_conn_enum ********************************************************************/ static bool api_srv_net_conn_enum(pipes_struct *p) { SRV_Q_NET_CONN_ENUM q_u; SRV_R_NET_CONN_ENUM r_u; prs_struct *data = &p->in_data.data; prs_struct *rdata = &p->out_data.rdata; ZERO_STRUCT(q_u); ZERO_STRUCT(r_u); /* grab the net server get enum */ if (!srv_io_q_net_conn_enum("", &q_u, data, 0)) return False; r_u.status = _srv_net_conn_enum(p, &q_u, &r_u); /* store the response in the SMB stream */ if (!srv_io_r_net_conn_enum("", &r_u, rdata, 0)) return False; return True; } /******************************************************************* Enumerate sessions. ********************************************************************/ static bool api_srv_net_sess_enum(pipes_struct *p) { SRV_Q_NET_SESS_ENUM q_u; SRV_R_NET_SESS_ENUM r_u; prs_struct *data = &p->in_data.data; prs_struct *rdata = &p->out_data.rdata; ZERO_STRUCT(q_u); ZERO_STRUCT(r_u); /* grab the net server get enum */ if (!srv_io_q_net_sess_enum("", &q_u, data, 0)) return False; /* construct reply. always indicate success */ r_u.status = _srv_net_sess_enum(p, &q_u, &r_u); /* store the response in the SMB stream */ if (!srv_io_r_net_sess_enum("", &r_u, rdata, 0)) return False; return True; } /******************************************************************* Delete session. ********************************************************************/ static bool api_srv_net_sess_del(pipes_struct *p) { SRV_Q_NET_SESS_DEL q_u; SRV_R_NET_SESS_DEL r_u; prs_struct *data = &p->in_data.data; prs_struct *rdata = &p->out_data.rdata; ZERO_STRUCT(q_u); ZERO_STRUCT(r_u); /* grab the net server get enum */ if (!srv_io_q_net_sess_del("", &q_u, data, 0)) return False; /* construct reply. always indicate success */ r_u.status = _srv_net_sess_del(p, &q_u, &r_u); /* store the response in the SMB stream */ if (!srv_io_r_net_sess_del("", &r_u, rdata, 0)) return False; return True; } /******************************************************************* RPC to enumerate shares. ********************************************************************/ static bool api_srv_net_share_enum_all(pipes_struct *p) { return proxy_srvsvc_call(p, NDR_SRVSVC_NETSHAREENUMALL); } /******************************************************************* RPC to enumerate shares. ********************************************************************/ static bool api_srv_net_share_enum(pipes_struct *p) { return proxy_srvsvc_call(p, NDR_SRVSVC_NETSHAREENUM); } /******************************************************************* RPC to return share information. ********************************************************************/ static bool api_srv_net_share_get_info(pipes_struct *p) { return proxy_srvsvc_call(p, NDR_SRVSVC_NETSHAREGETINFO); } /******************************************************************* RPC to set share information. ********************************************************************/ static bool api_srv_net_share_set_info(pipes_struct *p) { return proxy_srvsvc_call(p, NDR_SRVSVC_NETSHARESETINFO); } /******************************************************************* RPC to add share information. ********************************************************************/ static bool api_srv_net_share_add(pipes_struct *p) { return proxy_srvsvc_call(p, NDR_SRVSVC_NETSHAREADD); } /******************************************************************* RPC to delete share information. ********************************************************************/ static bool api_srv_net_share_del(pipes_struct *p) { return proxy_srvsvc_call(p, NDR_SRVSVC_NETSHAREDEL); } /******************************************************************* RPC to delete share information. ********************************************************************/ static bool api_srv_net_share_del_sticky(pipes_struct *p) { return proxy_srvsvc_call(p, NDR_SRVSVC_NETSHAREDELSTICKY); } /******************************************************************* api_srv_net_remote_tod ********************************************************************/ static bool api_srv_net_remote_tod(pipes_struct *p) { return proxy_srvsvc_call(p, NDR_SRVSVC_NETREMOTETOD); } /******************************************************************* RPC to enumerate disks available on a server e.g. C:, D: ... *******************************************************************/ static bool api_srv_net_disk_enum(pipes_struct *p) { SRV_Q_NET_DISK_ENUM q_u; SRV_R_NET_DISK_ENUM r_u; prs_struct *data = &p->in_data.data; prs_struct *rdata = &p->out_data.rdata; ZERO_STRUCT(q_u); ZERO_STRUCT(r_u); /* Unmarshall the net server disk enum. */ if(!srv_io_q_net_disk_enum("", &q_u, data, 0)) { DEBUG(0,("api_srv_net_disk_enum: Failed to unmarshall SRV_Q_NET_DISK_ENUM.\n")); return False; } r_u.status = _srv_net_disk_enum(p, &q_u, &r_u); if(!srv_io_r_net_disk_enum("", &r_u, rdata, 0)) { DEBUG(0,("api_srv_net_disk_enum: Failed to marshall SRV_R_NET_DISK_ENUM.\n")); return False; } return True; } /******************************************************************* NetValidateName (opnum 0x21) *******************************************************************/ static bool api_srv_net_name_validate(pipes_struct *p) { return proxy_srvsvc_call(p, NDR_SRVSVC_NETNAMEVALIDATE); } /******************************************************************* NetFileQuerySecdesc (opnum 0x27) *******************************************************************/ static bool api_srv_net_file_query_secdesc(pipes_struct *p) { SRV_Q_NET_FILE_QUERY_SECDESC q_u; SRV_R_NET_FILE_QUERY_SECDESC r_u; prs_struct *data = &p->in_data.data; prs_struct *rdata = &p->out_data.rdata; ZERO_STRUCT(q_u); ZERO_STRUCT(r_u); /* Unmarshall the net file get info from Win9x */ if(!srv_io_q_net_file_query_secdesc("", &q_u, data, 0)) { DEBUG(0,("api_srv_net_file_query_secdesc: Failed to unmarshall SRV_Q_NET_FILE_QUERY_SECDESC.\n")); return False; } r_u.status = _srv_net_file_query_secdesc(p, &q_u, &r_u); if(!srv_io_r_net_file_query_secdesc("", &r_u, rdata, 0)) { DEBUG(0,("api_srv_net_file_query_secdesc: Failed to marshall SRV_R_NET_FILE_QUERY_SECDESC.\n")); return False; } return True; } /******************************************************************* NetFileSetSecdesc (opnum 0x28) *******************************************************************/ static bool api_srv_net_file_set_secdesc(pipes_struct *p) { SRV_Q_NET_FILE_SET_SECDESC q_u; SRV_R_NET_FILE_SET_SECDESC r_u; prs_struct *data = &p->in_data.data; prs_struct *rdata = &p->out_data.rdata; ZERO_STRUCT(q_u); ZERO_STRUCT(r_u); /* Unmarshall the net file set info from Win9x */ if(!srv_io_q_net_file_set_secdesc("", &q_u, data, 0)) { DEBUG(0,("api_srv_net_file_set_secdesc: Failed to unmarshall SRV_Q_NET_FILE_SET_SECDESC.\n")); return False; } r_u.status = _srv_net_file_set_secdesc(p, &q_u, &r_u); if(!srv_io_r_net_file_set_secdesc("", &r_u, rdata, 0)) { DEBUG(0,("api_srv_net_file_set_secdesc: Failed to marshall SRV_R_NET_FILE_SET_SECDESC.\n")); return False; } return True; } /******************************************************************* *******************************************************************/ static bool api_srv_net_file_close(pipes_struct *p) { return proxy_srvsvc_call( p, NDR_SRVSVC_NETFILECLOSE ); } /******************************************************************* \PIPE\srvsvc commands ********************************************************************/ static struct api_struct api_srv_cmds[] = { { "SRV_NET_CONN_ENUM" , SRV_NET_CONN_ENUM , api_srv_net_conn_enum }, { "SRV_NET_SESS_ENUM" , SRV_NET_SESS_ENUM , api_srv_net_sess_enum }, { "SRV_NET_SESS_DEL" , SRV_NET_SESS_DEL , api_srv_net_sess_del }, { "SRV_NET_SHARE_ENUM_ALL" , SRV_NET_SHARE_ENUM_ALL , api_srv_net_share_enum_all }, { "SRV_NET_SHARE_ENUM" , SRV_NET_SHARE_ENUM , api_srv_net_share_enum }, { "SRV_NET_SHARE_ADD" , SRV_NET_SHARE_ADD , api_srv_net_share_add }, { "SRV_NET_SHARE_DEL" , SRV_NET_SHARE_DEL , api_srv_net_share_del }, { "SRV_NET_SHARE_DEL_STICKY" , SRV_NET_SHARE_DEL_STICKY , api_srv_net_share_del_sticky }, { "SRV_NET_SHARE_GET_INFO" , SRV_NET_SHARE_GET_INFO , api_srv_net_share_get_info }, { "SRV_NET_SHARE_SET_INFO" , SRV_NET_SHARE_SET_INFO , api_srv_net_share_set_info }, { "SRV_NET_FILE_ENUM" , SRV_NET_FILE_ENUM , api_srv_net_file_enum }, { "SRV_NET_SRV_GET_INFO" , SRV_NET_SRV_GET_INFO , api_srv_net_srv_get_info }, { "SRV_NET_SRV_SET_INFO" , SRV_NET_SRV_SET_INFO , api_srv_net_srv_set_info }, { "SRV_NET_REMOTE_TOD" , SRV_NET_REMOTE_TOD , api_srv_net_remote_tod }, { "SRV_NET_DISK_ENUM" , SRV_NET_DISK_ENUM , api_srv_net_disk_enum }, { "SRV_NET_NAME_VALIDATE" , SRV_NET_NAME_VALIDATE , api_srv_net_name_validate }, { "SRV_NET_FILE_QUERY_SECDESC", SRV_NET_FILE_QUERY_SECDESC, api_srv_net_file_query_secdesc }, { "SRV_NET_FILE_SET_SECDESC" , SRV_NET_FILE_SET_SECDESC , api_srv_net_file_set_secdesc }, { "SRV_NET_FILE_CLOSE" , SRV_NET_FILE_CLOSE , api_srv_net_file_close } }; void srvsvc2_get_pipe_fns( struct api_struct **fns, int *n_fns ) { *fns = api_srv_cmds; *n_fns = sizeof(api_srv_cmds) / sizeof(struct api_struct); } NTSTATUS rpc_srvsvc2_init(void) { return rpc_pipe_register_commands(SMB_RPC_INTERFACE_VERSION, "srvsvc", "ntsvcs", api_srv_cmds, sizeof(api_srv_cmds) / sizeof(struct api_struct)); }