/* * Unix SMB/CIFS implementation. * RPC Pipe client / server routines * Copyright (C) Gerald (Jerry) Carter 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 * 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 . */ #include "includes.h" #undef DBGC_CLASS #define DBGC_CLASS DBGC_RPC_PARSE /******************************************************************* ********************************************************************/ static bool svcctl_io_service_status( const char *desc, SERVICE_STATUS *status, prs_struct *ps, int depth ) { prs_debug(ps, depth, desc, "svcctl_io_service_status"); depth++; if(!prs_uint32("type", ps, depth, &status->type)) return False; if(!prs_uint32("state", ps, depth, &status->state)) return False; if(!prs_uint32("controls_accepted", ps, depth, &status->controls_accepted)) return False; if(!prs_werror("win32_exit_code", ps, depth, &status->win32_exit_code)) return False; if(!prs_uint32("service_exit_code", ps, depth, &status->service_exit_code)) return False; if(!prs_uint32("check_point", ps, depth, &status->check_point)) return False; if(!prs_uint32("wait_hint", ps, depth, &status->wait_hint)) return False; return True; } /******************************************************************* ********************************************************************/ static bool svcctl_io_service_config( const char *desc, SERVICE_CONFIG *config, prs_struct *ps, int depth ) { prs_debug(ps, depth, desc, "svcctl_io_service_config"); depth++; if(!prs_uint32("service_type", ps, depth, &config->service_type)) return False; if(!prs_uint32("start_type", ps, depth, &config->start_type)) return False; if(!prs_uint32("error_control", ps, depth, &config->error_control)) return False; if (!prs_io_unistr2_p("", ps, depth, &config->executablepath)) return False; if (!prs_io_unistr2_p("", ps, depth, &config->loadordergroup)) return False; if(!prs_uint32("tag_id", ps, depth, &config->tag_id)) return False; if (!prs_io_unistr2_p("", ps, depth, &config->dependencies)) return False; if (!prs_io_unistr2_p("", ps, depth, &config->startname)) return False; if (!prs_io_unistr2_p("", ps, depth, &config->displayname)) return False; if (!prs_io_unistr2("", ps, depth, config->executablepath)) return False; if (!prs_io_unistr2("", ps, depth, config->loadordergroup)) return False; if (!prs_io_unistr2("", ps, depth, config->dependencies)) return False; if (!prs_io_unistr2("", ps, depth, config->startname)) return False; if (!prs_io_unistr2("", ps, depth, config->displayname)) return False; return True; } /******************************************************************* ********************************************************************/ bool svcctl_io_enum_services_status( const char *desc, ENUM_SERVICES_STATUS *enum_status, RPC_BUFFER *buffer, int depth ) { prs_struct *ps=&buffer->prs; prs_debug(ps, depth, desc, "svcctl_io_enum_services_status"); depth++; if ( !smb_io_relstr("servicename", buffer, depth, &enum_status->servicename) ) return False; if ( !smb_io_relstr("displayname", buffer, depth, &enum_status->displayname) ) return False; if ( !svcctl_io_service_status("svc_status", &enum_status->status, ps, depth) ) return False; return True; } /******************************************************************* ********************************************************************/ bool svcctl_io_service_status_process( const char *desc, SERVICE_STATUS_PROCESS *status, RPC_BUFFER *buffer, int depth ) { prs_struct *ps=&buffer->prs; prs_debug(ps, depth, desc, "svcctl_io_service_status_process"); depth++; if ( !svcctl_io_service_status("status", &status->status, ps, depth) ) return False; if(!prs_align(ps)) return False; if(!prs_uint32("process_id", ps, depth, &status->process_id)) return False; if(!prs_uint32("service_flags", ps, depth, &status->service_flags)) return False; return True; } /******************************************************************* ********************************************************************/ uint32 svcctl_sizeof_enum_services_status( ENUM_SERVICES_STATUS *status ) { uint32 size = 0; size += size_of_relative_string( &status->servicename ); size += size_of_relative_string( &status->displayname ); size += sizeof(SERVICE_STATUS); return size; } /******************************************************************** ********************************************************************/ static uint32 sizeof_unistr2( UNISTR2 *string ) { uint32 size = 0; if ( !string ) return 0; size = sizeof(uint32) * 3; /* length fields */ size += 2 * string->uni_max_len; /* string data */ size += size % 4; /* alignment */ return size; } /******************************************************************** ********************************************************************/ uint32 svcctl_sizeof_service_config( SERVICE_CONFIG *config ) { uint32 size = 0; size = sizeof(uint32) * 4; /* static uint32 fields */ /* now add the UNISTR2 + pointer sizes */ size += sizeof(uint32) * sizeof_unistr2(config->executablepath); size += sizeof(uint32) * sizeof_unistr2(config->loadordergroup); size += sizeof(uint32) * sizeof_unistr2(config->dependencies); size += sizeof(uint32) * sizeof_unistr2(config->startname); size += sizeof(uint32) * sizeof_unistr2(config->displayname); return size; } /******************************************************************* ********************************************************************/ bool svcctl_io_q_query_status(const char *desc, SVCCTL_Q_QUERY_STATUS *q_u, prs_struct *ps, int depth) { if (q_u == NULL) return False; prs_debug(ps, depth, desc, "svcctl_io_q_query_status"); depth++; if(!prs_align(ps)) return False; if(!smb_io_pol_hnd("service_pol", &q_u->handle, ps, depth)) return False; return True; } /******************************************************************* ********************************************************************/ bool svcctl_io_r_query_status(const char *desc, SVCCTL_R_QUERY_STATUS *r_u, prs_struct *ps, int depth) { if (r_u == NULL) return False; prs_debug(ps, depth, desc, "svcctl_io_r_query_status"); depth++; if(!prs_align(ps)) return False; if(!svcctl_io_service_status("service_status", &r_u->svc_status, ps, depth)) return False; if(!prs_werror("status", ps, depth, &r_u->status)) return False; return True; } /******************************************************************* ********************************************************************/ bool svcctl_io_q_enum_services_status(const char *desc, SVCCTL_Q_ENUM_SERVICES_STATUS *q_u, prs_struct *ps, int depth) { if (q_u == NULL) return False; prs_debug(ps, depth, desc, "svcctl_io_q_enum_services_status"); depth++; if(!prs_align(ps)) return False; if(!smb_io_pol_hnd("scm_pol", &q_u->handle, ps, depth)) return False; if(!prs_uint32("type", ps, depth, &q_u->type)) return False; if(!prs_uint32("state", ps, depth, &q_u->state)) return False; if(!prs_uint32("buffer_size", ps, depth, &q_u->buffer_size)) return False; if(!prs_pointer("resume", ps, depth, (void*)&q_u->resume, sizeof(uint32), (PRS_POINTER_CAST)prs_uint32)) return False; return True; } /******************************************************************* ********************************************************************/ bool svcctl_io_r_enum_services_status(const char *desc, SVCCTL_R_ENUM_SERVICES_STATUS *r_u, prs_struct *ps, int depth) { if (r_u == NULL) return False; prs_debug(ps, depth, desc, "svcctl_io_r_enum_services_status"); depth++; if(!prs_align(ps)) return False; if (!prs_rpcbuffer("", ps, depth, &r_u->buffer)) return False; if(!prs_align(ps)) return False; if(!prs_uint32("needed", ps, depth, &r_u->needed)) return False; if(!prs_uint32("returned", ps, depth, &r_u->returned)) return False; if(!prs_pointer("resume", ps, depth, (void*)&r_u->resume, sizeof(uint32), (PRS_POINTER_CAST)prs_uint32)) return False; if(!prs_werror("status", ps, depth, &r_u->status)) return False; return True; } /******************************************************************* ********************************************************************/ bool svcctl_io_q_enum_dependent_services(const char *desc, SVCCTL_Q_ENUM_DEPENDENT_SERVICES *q_u, prs_struct *ps, int depth) { if (q_u == NULL) return False; prs_debug(ps, depth, desc, "svcctl_io_q_enum_dependent_services"); depth++; if(!prs_align(ps)) return False; if(!smb_io_pol_hnd("service_pol", &q_u->handle, ps, depth)) return False; if(!prs_uint32("state", ps, depth, &q_u->state)) return False; if(!prs_uint32("buffer_size", ps, depth, &q_u->buffer_size)) return False; return True; } /******************************************************************* ********************************************************************/ bool svcctl_io_r_enum_dependent_services(const char *desc, SVCCTL_R_ENUM_DEPENDENT_SERVICES *r_u, prs_struct *ps, int depth) { if (r_u == NULL) return False; prs_debug(ps, depth, desc, "svcctl_io_r_enum_dependent_services"); depth++; if(!prs_align(ps)) return False; if (!prs_rpcbuffer("", ps, depth, &r_u->buffer)) return False; if(!prs_align(ps)) return False; if(!prs_uint32("needed", ps, depth, &r_u->needed)) return False; if(!prs_uint32("returned", ps, depth, &r_u->returned)) return False; if(!prs_werror("status", ps, depth, &r_u->status)) return False; return True; } /******************************************************************* ********************************************************************/ bool svcctl_io_q_control_service(const char *desc, SVCCTL_Q_CONTROL_SERVICE *q_u, prs_struct *ps, int depth) { if (q_u == NULL) return False; prs_debug(ps, depth, desc, "svcctl_io_q_control_service"); depth++; if(!prs_align(ps)) return False; if(!smb_io_pol_hnd("service_pol", &q_u->handle, ps, depth)) return False; if(!prs_uint32("control", ps, depth, &q_u->control)) return False; return True; } /******************************************************************* ********************************************************************/ bool svcctl_io_r_control_service(const char *desc, SVCCTL_R_CONTROL_SERVICE *r_u, prs_struct *ps, int depth) { if (r_u == NULL) return False; prs_debug(ps, depth, desc, "svcctl_io_r_control_service"); depth++; if(!prs_align(ps)) return False; if(!svcctl_io_service_status("service_status", &r_u->svc_status, ps, depth)) return False; if(!prs_werror("status", ps, depth, &r_u->status)) return False; return True; } /******************************************************************* ********************************************************************/ bool svcctl_io_q_query_service_config(const char *desc, SVCCTL_Q_QUERY_SERVICE_CONFIG *q_u, prs_struct *ps, int depth) { if (q_u == NULL) return False; prs_debug(ps, depth, desc, "svcctl_io_q_query_service_config"); depth++; if(!prs_align(ps)) return False; if(!smb_io_pol_hnd("service_pol", &q_u->handle, ps, depth)) return False; if(!prs_uint32("buffer_size", ps, depth, &q_u->buffer_size)) return False; return True; } /******************************************************************* ********************************************************************/ bool svcctl_io_r_query_service_config(const char *desc, SVCCTL_R_QUERY_SERVICE_CONFIG *r_u, prs_struct *ps, int depth) { if (r_u == NULL) return False; prs_debug(ps, depth, desc, "svcctl_io_r_query_service_config"); depth++; if(!prs_align(ps)) return False; if(!svcctl_io_service_config("config", &r_u->config, ps, depth)) return False; if(!prs_uint32("needed", ps, depth, &r_u->needed)) return False; if(!prs_werror("status", ps, depth, &r_u->status)) return False; return True; } /******************************************************************* ********************************************************************/ bool svcctl_io_q_query_service_config2(const char *desc, SVCCTL_Q_QUERY_SERVICE_CONFIG2 *q_u, prs_struct *ps, int depth) { if (q_u == NULL) return False; prs_debug(ps, depth, desc, "svcctl_io_q_query_service_config2"); depth++; if(!prs_align(ps)) return False; if(!smb_io_pol_hnd("service_pol", &q_u->handle, ps, depth)) return False; if(!prs_uint32("level", ps, depth, &q_u->level)) return False; if(!prs_uint32("buffer_size", ps, depth, &q_u->buffer_size)) return False; return True; } /******************************************************************* ********************************************************************/ void init_service_description_buffer(SERVICE_DESCRIPTION *desc, const char *service_desc ) { desc->unknown = 0x04; /* always 0x0000 0004 (no idea what this is) */ init_unistr( &desc->description, service_desc ); } /******************************************************************* ********************************************************************/ bool svcctl_io_service_description( const char *desc, SERVICE_DESCRIPTION *description, RPC_BUFFER *buffer, int depth ) { prs_struct *ps = &buffer->prs; prs_debug(ps, depth, desc, "svcctl_io_service_description"); depth++; if ( !prs_uint32("unknown", ps, depth, &description->unknown) ) return False; if ( !prs_unistr("description", ps, depth, &description->description) ) return False; return True; } /******************************************************************* ********************************************************************/ uint32 svcctl_sizeof_service_description( SERVICE_DESCRIPTION *desc ) { if ( !desc ) return 0; /* make sure to include the terminating NULL */ return ( sizeof(uint32) + (2*(str_len_uni(&desc->description)+1)) ); } /******************************************************************* ********************************************************************/ static bool svcctl_io_action( const char *desc, SC_ACTION *action, prs_struct *ps, int depth ) { prs_debug(ps, depth, desc, "svcctl_io_action"); depth++; if ( !prs_uint32("type", ps, depth, &action->type) ) return False; if ( !prs_uint32("delay", ps, depth, &action->delay) ) return False; return True; } /******************************************************************* ********************************************************************/ bool svcctl_io_service_fa( const char *desc, SERVICE_FAILURE_ACTIONS *fa, RPC_BUFFER *buffer, int depth ) { prs_struct *ps = &buffer->prs; int i; prs_debug(ps, depth, desc, "svcctl_io_service_description"); depth++; if ( !prs_uint32("reset_period", ps, depth, &fa->reset_period) ) return False; if ( !prs_pointer( desc, ps, depth, (void*)&fa->rebootmsg, sizeof(UNISTR2), (PRS_POINTER_CAST)prs_io_unistr2 ) ) return False; if ( !prs_pointer( desc, ps, depth, (void*)&fa->command, sizeof(UNISTR2), (PRS_POINTER_CAST)prs_io_unistr2 ) ) return False; if ( !prs_uint32("num_actions", ps, depth, &fa->num_actions) ) return False; if ( UNMARSHALLING(ps)) { if (fa->num_actions) { if ( !(fa->actions = TALLOC_ARRAY( talloc_tos(), SC_ACTION, fa->num_actions )) ) { DEBUG(0,("svcctl_io_service_fa: talloc() failure!\n")); return False; } } else { fa->actions = NULL; } } for ( i=0; inum_actions; i++ ) { if ( !svcctl_io_action( "actions", &fa->actions[i], ps, depth ) ) return False; } return True; } /******************************************************************* ********************************************************************/ uint32 svcctl_sizeof_service_fa( SERVICE_FAILURE_ACTIONS *fa) { uint32 size = 0; if ( !fa ) return 0; size = sizeof(uint32) * 2; size += sizeof_unistr2( fa->rebootmsg ); size += sizeof_unistr2( fa->command ); size += sizeof(SC_ACTION) * fa->num_actions; return size; } /******************************************************************* ********************************************************************/ bool svcctl_io_r_query_service_config2(const char *desc, SVCCTL_R_QUERY_SERVICE_CONFIG2 *r_u, prs_struct *ps, int depth) { if ( !r_u ) return False; prs_debug(ps, depth, desc, "svcctl_io_r_query_service_config2"); depth++; if ( !prs_align(ps) ) return False; if (!prs_rpcbuffer("", ps, depth, &r_u->buffer)) return False; if(!prs_align(ps)) return False; if (!prs_uint32("needed", ps, depth, &r_u->needed)) return False; if(!prs_werror("status", ps, depth, &r_u->status)) return False; return True; } /******************************************************************* ********************************************************************/ bool svcctl_io_q_query_service_status_ex(const char *desc, SVCCTL_Q_QUERY_SERVICE_STATUSEX *q_u, prs_struct *ps, int depth) { if (q_u == NULL) return False; prs_debug(ps, depth, desc, "svcctl_io_q_query_service_status_ex"); depth++; if(!prs_align(ps)) return False; if(!smb_io_pol_hnd("service_pol", &q_u->handle, ps, depth)) return False; if(!prs_uint32("level", ps, depth, &q_u->level)) return False; if(!prs_uint32("buffer_size", ps, depth, &q_u->buffer_size)) return False; return True; } /******************************************************************* ********************************************************************/ bool svcctl_io_r_query_service_status_ex(const char *desc, SVCCTL_R_QUERY_SERVICE_STATUSEX *r_u, prs_struct *ps, int depth) { if ( !r_u ) return False; prs_debug(ps, depth, desc, "svcctl_io_r_query_service_status_ex"); depth++; if (!prs_rpcbuffer("", ps, depth, &r_u->buffer)) return False; if(!prs_align(ps)) return False; if(!prs_uint32("needed", ps, depth, &r_u->needed)) return False; if(!prs_werror("status", ps, depth, &r_u->status)) return False; return True; } /******************************************************************* ********************************************************************/ bool svcctl_io_q_lock_service_db(const char *desc, SVCCTL_Q_LOCK_SERVICE_DB *q_u, prs_struct *ps, int depth) { if (q_u == NULL) return False; prs_debug(ps, depth, desc, "svcctl_io_q_lock_service_db"); depth++; if(!prs_align(ps)) return False; if(!smb_io_pol_hnd("scm_handle", &q_u->handle, ps, depth)) return False; return True; } /******************************************************************* ********************************************************************/ bool svcctl_io_r_lock_service_db(const char *desc, SVCCTL_R_LOCK_SERVICE_DB *r_u, prs_struct *ps, int depth) { if ( !r_u ) return False; prs_debug(ps, depth, desc, "svcctl_io_r_lock_service_db"); depth++; if(!prs_align(ps)) return False; if(!smb_io_pol_hnd("lock_handle", &r_u->h_lock, ps, depth)) return False; if(!prs_werror("status", ps, depth, &r_u->status)) return False; return True; } /******************************************************************* ********************************************************************/ bool svcctl_io_q_unlock_service_db(const char *desc, SVCCTL_Q_UNLOCK_SERVICE_DB *q_u, prs_struct *ps, int depth) { if (q_u == NULL) return False; prs_debug(ps, depth, desc, "svcctl_io_q_unlock_service_db"); depth++; if(!prs_align(ps)) return False; if(!smb_io_pol_hnd("h_lock", &q_u->h_lock, ps, depth)) return False; return True; } /******************************************************************* ********************************************************************/ bool svcctl_io_r_unlock_service_db(const char *desc, SVCCTL_R_UNLOCK_SERVICE_DB *r_u, prs_struct *ps, int depth) { if ( !r_u ) return False; prs_debug(ps, depth, desc, "svcctl_io_r_unlock_service_db"); depth++; if(!prs_align(ps)) return False; if(!prs_werror("status", ps, depth, &r_u->status)) return False; return True; } /******************************************************************* ********************************************************************/ bool svcctl_io_q_query_service_sec(const char *desc, SVCCTL_Q_QUERY_SERVICE_SEC *q_u, prs_struct *ps, int depth) { if (q_u == NULL) return False; prs_debug(ps, depth, desc, "svcctl_io_q_query_service_sec"); depth++; if(!prs_align(ps)) return False; if(!smb_io_pol_hnd("handle", &q_u->handle, ps, depth)) return False; if(!prs_uint32("security_flags", ps, depth, &q_u->security_flags)) return False; if(!prs_uint32("buffer_size", ps, depth, &q_u->buffer_size)) return False; return True; } /******************************************************************* ********************************************************************/ bool svcctl_io_r_query_service_sec(const char *desc, SVCCTL_R_QUERY_SERVICE_SEC *r_u, prs_struct *ps, int depth) { if ( !r_u ) return False; prs_debug(ps, depth, desc, "svcctl_io_r_query_service_sec"); depth++; if(!prs_align(ps)) return False; if (!prs_rpcbuffer("buffer", ps, depth, &r_u->buffer)) return False; if(!prs_uint32("needed", ps, depth, &r_u->needed)) return False; if(!prs_werror("status", ps, depth, &r_u->status)) return False; return True; } /******************************************************************* ********************************************************************/ bool svcctl_io_q_set_service_sec(const char *desc, SVCCTL_Q_SET_SERVICE_SEC *q_u, prs_struct *ps, int depth) { if (q_u == NULL) return False; prs_debug(ps, depth, desc, "svcctl_io_q_set_service_sec"); depth++; if(!prs_align(ps)) return False; if(!smb_io_pol_hnd("handle", &q_u->handle, ps, depth)) return False; if(!prs_uint32("security_flags", ps, depth, &q_u->security_flags)) return False; if (!prs_rpcbuffer("buffer", ps, depth, &q_u->buffer)) return False; if(!prs_align(ps)) return False; if(!prs_uint32("buffer_size", ps, depth, &q_u->buffer_size)) return False; return True; } /******************************************************************* ********************************************************************/ bool svcctl_io_r_set_service_sec(const char *desc, SVCCTL_R_SET_SERVICE_SEC *r_u, prs_struct *ps, int depth) { if ( !r_u ) return False; prs_debug(ps, depth, desc, "svcctl_io_r_set_service_sec"); depth++; if(!prs_align(ps)) return False; if(!prs_werror("status", ps, depth, &r_u->status)) return False; return True; }