2000-04-19 11:00:22 +04:00
/*
Unix SMB / Netbios implementation .
Version 1.9 .
Inter - process communication and named pipe handling
Copyright ( C ) Andrew Tridgell 1992 - 1998
SMB Version handling
Copyright ( C ) John H Terpstra 1995 - 1998
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 2 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 , write to the Free Software
Foundation , Inc . , 675 Mass Ave , Cambridge , MA 0213 9 , USA .
*/
/*
This file handles the named pipe and mailslot calls
in the SMBtrans protocol
*/
# include "includes.h"
# ifdef CHECK_TYPES
# undef CHECK_TYPES
# endif
# define CHECK_TYPES 0
extern int DEBUGLEVEL ;
extern fstring local_machine ;
2000-10-06 08:36:05 +04:00
extern pstring global_myname ;
2000-04-19 11:00:22 +04:00
extern fstring global_myworkgroup ;
# define NERR_Success 0
# define NERR_badpass 86
# define NERR_notsupported 50
# define NERR_BASE (2100)
# define NERR_BufTooSmall (NERR_BASE+23)
# define NERR_JobNotFound (NERR_BASE+51)
# define NERR_DestNotFound (NERR_BASE+52)
# define ACCESS_READ 0x01
# define ACCESS_WRITE 0x02
# define ACCESS_CREATE 0x04
# define SHPWLEN 8 /* share password length */
static BOOL api_Unsupported ( connection_struct * conn , uint16 vuid , char * param , char * data ,
int mdrcnt , int mprcnt ,
char * * rdata , char * * rparam ,
int * rdata_len , int * rparam_len ) ;
static BOOL api_TooSmall ( connection_struct * conn , uint16 vuid , char * param , char * data ,
int mdrcnt , int mprcnt ,
char * * rdata , char * * rparam ,
int * rdata_len , int * rparam_len ) ;
static int CopyExpanded ( connection_struct * conn ,
int snum , char * * dst , char * src , int * n )
{
pstring buf ;
int l ;
if ( ! src | | ! dst | | ! n | | ! ( * dst ) ) return ( 0 ) ;
StrnCpy ( buf , src , sizeof ( buf ) / 2 ) ;
pstring_sub ( buf , " %S " , lp_servicename ( snum ) ) ;
standard_sub_conn ( conn , buf ) ;
2001-07-04 11:15:53 +04:00
l = push_ascii ( * dst , buf , * n - 1 , STR_TERMINATE ) ;
2000-04-19 11:00:22 +04:00
( * dst ) + = l ;
( * n ) - = l ;
return l ;
}
static int CopyAndAdvance ( char * * dst , char * src , int * n )
{
int l ;
if ( ! src | | ! dst | | ! n | | ! ( * dst ) ) return ( 0 ) ;
2001-07-04 11:15:53 +04:00
l = push_ascii ( * dst , src , * n - 1 , STR_TERMINATE ) ;
2000-04-19 11:00:22 +04:00
( * dst ) + = l ;
( * n ) - = l ;
return l ;
}
static int StrlenExpanded ( connection_struct * conn , int snum , char * s )
{
pstring buf ;
if ( ! s ) return ( 0 ) ;
StrnCpy ( buf , s , sizeof ( buf ) / 2 ) ;
pstring_sub ( buf , " %S " , lp_servicename ( snum ) ) ;
standard_sub_conn ( conn , buf ) ;
return strlen ( buf ) + 1 ;
}
static char * Expand ( connection_struct * conn , int snum , char * s )
{
static pstring buf ;
if ( ! s ) return ( NULL ) ;
StrnCpy ( buf , s , sizeof ( buf ) / 2 ) ;
pstring_sub ( buf , " %S " , lp_servicename ( snum ) ) ;
standard_sub_conn ( conn , buf ) ;
return & buf [ 0 ] ;
}
/*******************************************************************
check a API string for validity when we only need to check the prefix
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static BOOL prefix_ok ( char * str , char * prefix )
{
return ( strncmp ( str , prefix , strlen ( prefix ) ) = = 0 ) ;
}
struct pack_desc {
char * format ; /* formatstring for structure */
char * subformat ; /* subformat for structure */
char * base ; /* baseaddress of buffer */
int buflen ; /* remaining size for fixed part; on init: length of base */
int subcount ; /* count of substructures */
char * structbuf ; /* pointer into buffer for remaining fixed part */
int stringlen ; /* remaining size for variable part */
char * stringbuf ; /* pointer into buffer for remaining variable part */
int neededlen ; /* total needed size */
int usedlen ; /* total used size (usedlen <= neededlen and usedlen <= buflen) */
char * curpos ; /* current position; pointer into format or subformat */
int errcode ;
} ;
static int get_counter ( char * * p )
{
int i , n ;
if ( ! p | | ! ( * p ) ) return ( 1 ) ;
if ( ! isdigit ( ( int ) * * p ) ) return 1 ;
for ( n = 0 ; ; ) {
i = * * p ;
if ( isdigit ( i ) )
n = 10 * n + ( i - ' 0 ' ) ;
else
return n ;
( * p ) + + ;
}
}
static int getlen ( char * p )
{
int n = 0 ;
if ( ! p ) return ( 0 ) ;
while ( * p ) {
switch ( * p + + ) {
case ' W ' : /* word (2 byte) */
n + = 2 ;
break ;
case ' K ' : /* status word? (2 byte) */
n + = 2 ;
break ;
case ' N ' : /* count of substructures (word) at end */
n + = 2 ;
break ;
case ' D ' : /* double word (4 byte) */
case ' z ' : /* offset to zero terminated string (4 byte) */
case ' l ' : /* offset to user data (4 byte) */
n + = 4 ;
break ;
case ' b ' : /* offset to data (with counter) (4 byte) */
n + = 4 ;
get_counter ( & p ) ;
break ;
case ' B ' : /* byte (with optional counter) */
n + = get_counter ( & p ) ;
break ;
}
}
return n ;
}
static BOOL init_package ( struct pack_desc * p , int count , int subcount )
{
int n = p - > buflen ;
int i ;
if ( ! p - > format | | ! p - > base ) return ( False ) ;
i = count * getlen ( p - > format ) ;
if ( p - > subformat ) i + = subcount * getlen ( p - > subformat ) ;
p - > structbuf = p - > base ;
p - > neededlen = 0 ;
p - > usedlen = 0 ;
p - > subcount = 0 ;
p - > curpos = p - > format ;
if ( i > n ) {
p - > neededlen = i ;
i = n = 0 ;
2000-11-16 22:08:55 +03:00
#if 0
/*
* This is the old error code we used . Aparently
* WinNT / 2 k systems return ERRbuftoosmall ( 2123 ) and
* OS / 2 needs this . I ' m leaving this here so we can revert
* if needed . JRA .
*/
2000-04-19 11:00:22 +04:00
p - > errcode = ERRmoredata ;
2000-11-16 22:08:55 +03:00
# else
p - > errcode = ERRbuftoosmall ;
# endif
2000-04-19 11:00:22 +04:00
}
else
p - > errcode = NERR_Success ;
p - > buflen = i ;
n - = i ;
p - > stringbuf = p - > base + i ;
p - > stringlen = n ;
return ( p - > errcode = = NERR_Success ) ;
}
static int package ( struct pack_desc * p , . . . )
{
va_list args ;
int needed = 0 , stringneeded ;
char * str = NULL ;
int is_string = 0 , stringused ;
int32 temp ;
va_start ( args , p ) ;
if ( ! * p - > curpos ) {
if ( ! p - > subcount )
p - > curpos = p - > format ;
else {
p - > curpos = p - > subformat ;
p - > subcount - - ;
}
}
# if CHECK_TYPES
str = va_arg ( args , char * ) ;
SMB_ASSERT ( strncmp ( str , p - > curpos , strlen ( str ) ) = = 0 ) ;
# endif
stringneeded = - 1 ;
if ( ! p - > curpos ) {
va_end ( args ) ;
return ( 0 ) ;
}
switch ( * p - > curpos + + ) {
case ' W ' : /* word (2 byte) */
needed = 2 ;
temp = va_arg ( args , int ) ;
if ( p - > buflen > = needed ) SSVAL ( p - > structbuf , 0 , temp ) ;
break ;
case ' K ' : /* status word? (2 byte) */
needed = 2 ;
temp = va_arg ( args , int ) ;
if ( p - > buflen > = needed ) SSVAL ( p - > structbuf , 0 , temp ) ;
break ;
case ' N ' : /* count of substructures (word) at end */
needed = 2 ;
p - > subcount = va_arg ( args , int ) ;
if ( p - > buflen > = needed ) SSVAL ( p - > structbuf , 0 , p - > subcount ) ;
break ;
case ' D ' : /* double word (4 byte) */
needed = 4 ;
temp = va_arg ( args , int ) ;
if ( p - > buflen > = needed ) SIVAL ( p - > structbuf , 0 , temp ) ;
break ;
case ' B ' : /* byte (with optional counter) */
needed = get_counter ( & p - > curpos ) ;
{
char * s = va_arg ( args , char * ) ;
if ( p - > buflen > = needed ) StrnCpy ( p - > structbuf , s ? s : " " , needed - 1 ) ;
}
break ;
case ' z ' : /* offset to zero terminated string (4 byte) */
str = va_arg ( args , char * ) ;
stringneeded = ( str ? strlen ( str ) + 1 : 0 ) ;
is_string = 1 ;
break ;
case ' l ' : /* offset to user data (4 byte) */
str = va_arg ( args , char * ) ;
stringneeded = va_arg ( args , int ) ;
is_string = 0 ;
break ;
case ' b ' : /* offset to data (with counter) (4 byte) */
str = va_arg ( args , char * ) ;
stringneeded = get_counter ( & p - > curpos ) ;
is_string = 0 ;
break ;
}
va_end ( args ) ;
if ( stringneeded > = 0 ) {
needed = 4 ;
if ( p - > buflen > = needed ) {
stringused = stringneeded ;
if ( stringused > p - > stringlen ) {
stringused = ( is_string ? p - > stringlen : 0 ) ;
if ( p - > errcode = = NERR_Success ) p - > errcode = ERRmoredata ;
}
if ( ! stringused )
SIVAL ( p - > structbuf , 0 , 0 ) ;
else {
SIVAL ( p - > structbuf , 0 , PTR_DIFF ( p - > stringbuf , p - > base ) ) ;
memcpy ( p - > stringbuf , str ? str : " " , stringused ) ;
if ( is_string ) p - > stringbuf [ stringused - 1 ] = ' \0 ' ;
p - > stringbuf + = stringused ;
p - > stringlen - = stringused ;
p - > usedlen + = stringused ;
}
}
p - > neededlen + = stringneeded ;
}
p - > neededlen + = needed ;
if ( p - > buflen > = needed ) {
p - > structbuf + = needed ;
p - > buflen - = needed ;
p - > usedlen + = needed ;
}
else {
if ( p - > errcode = = NERR_Success ) p - > errcode = ERRmoredata ;
}
return 1 ;
}
# if CHECK_TYPES
# define PACK(desc,t,v) package(desc,t,v,0,0,0,0)
# define PACKl(desc,t,v,l) package(desc,t,v,l,0,0,0,0)
# else
# define PACK(desc,t,v) package(desc,v)
# define PACKl(desc,t,v,l) package(desc,v,l)
# endif
static void PACKI ( struct pack_desc * desc , char * t , int v )
{
PACK ( desc , t , v ) ;
}
static void PACKS ( struct pack_desc * desc , char * t , char * v )
{
PACK ( desc , t , v ) ;
}
/****************************************************************************
get a print queue
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static void PackDriverData ( struct pack_desc * desc )
{
char drivdata [ 4 + 4 + 32 ] ;
SIVAL ( drivdata , 0 , sizeof drivdata ) ; /* cb */
SIVAL ( drivdata , 4 , 1000 ) ; /* lVersion */
memset ( drivdata + 8 , 0 , 32 ) ; /* szDeviceName */
2001-07-04 11:15:53 +04:00
push_ascii ( drivdata + 8 , " NULL " , - 1 , STR_TERMINATE ) ;
2000-04-19 11:00:22 +04:00
PACKl ( desc , " l " , drivdata , sizeof drivdata ) ; /* pDriverData */
}
static int check_printq_info ( struct pack_desc * desc ,
int uLevel , char * id1 , char * id2 )
{
desc - > subformat = NULL ;
switch ( uLevel ) {
case 0 :
desc - > format = " B13 " ;
break ;
case 1 :
desc - > format = " B13BWWWzzzzzWW " ;
break ;
case 2 :
desc - > format = " B13BWWWzzzzzWN " ;
desc - > subformat = " WB21BB16B10zWWzDDz " ;
break ;
case 3 :
desc - > format = " zWWWWzzzzWWzzl " ;
break ;
case 4 :
desc - > format = " zWWWWzzzzWNzzl " ;
desc - > subformat = " WWzWWDDzz " ;
break ;
case 5 :
desc - > format = " z " ;
break ;
case 51 :
desc - > format = " K " ;
break ;
case 52 :
desc - > format = " WzzzzzzzzN " ;
desc - > subformat = " z " ;
break ;
default : return False ;
}
if ( strcmp ( desc - > format , id1 ) ! = 0 ) return False ;
if ( desc - > subformat & & strcmp ( desc - > subformat , id2 ) ! = 0 ) return False ;
return True ;
}
2001-02-23 06:59:37 +03:00
# define RAP_JOB_STATUS_QUEUED 0
# define RAP_JOB_STATUS_PAUSED 1
# define RAP_JOB_STATUS_SPOOLING 2
# define RAP_JOB_STATUS_PRINTING 3
# define RAP_JOB_STATUS_PRINTED 4
2000-04-19 11:00:22 +04:00
2001-02-23 06:59:37 +03:00
# define RAP_QUEUE_STATUS_PAUSED 1
# define RAP_QUEUE_STATUS_ERROR 2
2000-04-19 11:00:22 +04:00
/* turn a print job status into a on the wire status
*/
static int printj_status ( int v )
{
switch ( v ) {
case LPQ_QUEUED :
2001-02-23 06:59:37 +03:00
return RAP_JOB_STATUS_QUEUED ;
2000-04-19 11:00:22 +04:00
case LPQ_PAUSED :
2001-02-23 06:59:37 +03:00
return RAP_JOB_STATUS_PAUSED ;
2000-04-19 11:00:22 +04:00
case LPQ_SPOOLING :
2001-02-23 06:59:37 +03:00
return RAP_JOB_STATUS_SPOOLING ;
2000-04-19 11:00:22 +04:00
case LPQ_PRINTING :
2001-02-23 06:59:37 +03:00
return RAP_JOB_STATUS_PRINTING ;
2000-04-19 11:00:22 +04:00
}
return 0 ;
}
/* turn a print queue status into a on the wire status
*/
static int printq_status ( int v )
{
switch ( v ) {
case LPQ_QUEUED :
return 0 ;
case LPQ_PAUSED :
2001-02-23 06:59:37 +03:00
return RAP_QUEUE_STATUS_PAUSED ;
2000-04-19 11:00:22 +04:00
}
2001-02-23 06:59:37 +03:00
return RAP_QUEUE_STATUS_ERROR ;
2000-04-19 11:00:22 +04:00
}
static void fill_printjob_info ( connection_struct * conn , int snum , int uLevel ,
struct pack_desc * desc ,
print_queue_struct * queue , int n )
{
time_t t = queue - > time ;
/* the client expects localtime */
t - = TimeDiff ( t ) ;
PACKI ( desc , " W " , queue - > job ) ; /* uJobId */
if ( uLevel = = 1 ) {
PACKS ( desc , " B21 " , queue - > user ) ; /* szUserName */
PACKS ( desc , " B " , " " ) ; /* pad */
PACKS ( desc , " B16 " , " " ) ; /* szNotifyName */
PACKS ( desc , " B10 " , " PM_Q_RAW " ) ; /* szDataType */
PACKS ( desc , " z " , " " ) ; /* pszParms */
PACKI ( desc , " W " , n + 1 ) ; /* uPosition */
PACKI ( desc , " W " , printj_status ( queue - > status ) ) ; /* fsStatus */
PACKS ( desc , " z " , " " ) ; /* pszStatus */
PACKI ( desc , " D " , t ) ; /* ulSubmitted */
PACKI ( desc , " D " , queue - > size ) ; /* ulSize */
PACKS ( desc , " z " , queue - > file ) ; /* pszComment */
}
2001-03-23 06:12:58 +03:00
if ( uLevel = = 2 | | uLevel = = 3 | | uLevel = = 4 ) {
2000-04-19 11:00:22 +04:00
PACKI ( desc , " W " , queue - > priority ) ; /* uPriority */
PACKS ( desc , " z " , queue - > user ) ; /* pszUserName */
PACKI ( desc , " W " , n + 1 ) ; /* uPosition */
PACKI ( desc , " W " , printj_status ( queue - > status ) ) ; /* fsStatus */
PACKI ( desc , " D " , t ) ; /* ulSubmitted */
PACKI ( desc , " D " , queue - > size ) ; /* ulSize */
PACKS ( desc , " z " , " Samba " ) ; /* pszComment */
PACKS ( desc , " z " , queue - > file ) ; /* pszDocument */
if ( uLevel = = 3 ) {
PACKS ( desc , " z " , " " ) ; /* pszNotifyName */
PACKS ( desc , " z " , " PM_Q_RAW " ) ; /* pszDataType */
PACKS ( desc , " z " , " " ) ; /* pszParms */
PACKS ( desc , " z " , " " ) ; /* pszStatus */
PACKS ( desc , " z " , SERVICE ( snum ) ) ; /* pszQueue */
PACKS ( desc , " z " , " lpd " ) ; /* pszQProcName */
PACKS ( desc , " z " , " " ) ; /* pszQProcParms */
PACKS ( desc , " z " , " NULL " ) ; /* pszDriverName */
PackDriverData ( desc ) ; /* pDriverData */
PACKS ( desc , " z " , " " ) ; /* pszPrinterName */
2001-03-23 06:12:58 +03:00
} else if ( uLevel = = 4 ) { /* OS2 */
PACKS ( desc , " z " , " " ) ; /* pszSpoolFileName */
PACKS ( desc , " z " , " " ) ; /* pszPortName */
PACKS ( desc , " z " , " " ) ; /* pszStatus */
PACKI ( desc , " D " , 0 ) ; /* ulPagesSpooled */
PACKI ( desc , " D " , 0 ) ; /* ulPagesSent */
PACKI ( desc , " D " , 0 ) ; /* ulPagesPrinted */
PACKI ( desc , " D " , 0 ) ; /* ulTimePrinted */
PACKI ( desc , " D " , 0 ) ; /* ulExtendJobStatus */
PACKI ( desc , " D " , 0 ) ; /* ulStartPage */
PACKI ( desc , " D " , 0 ) ; /* ulEndPage */
2000-04-19 11:00:22 +04:00
}
}
}
2000-12-08 00:55:10 +03:00
/********************************************************************
Return a driver name given an snum .
Looks in a tdb first . Returns True if from tdb , False otherwise .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static BOOL get_driver_name ( int snum , pstring drivername )
{
NT_PRINTER_INFO_LEVEL * info = NULL ;
BOOL in_tdb = False ;
get_a_printer ( & info , 2 , lp_servicename ( snum ) ) ;
if ( info ! = NULL ) {
pstrcpy ( drivername , info - > info_2 - > drivername ) ;
in_tdb = True ;
free_a_printer ( & info , 2 ) ;
} else {
pstrcpy ( drivername , lp_printerdriver ( snum ) ) ;
}
return in_tdb ;
}
2000-08-18 00:05:05 +04:00
/********************************************************************
Respond to the DosPrintQInfo command with a level of 52
This is used to get printer driver information for Win9x clients
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2000-04-19 11:00:22 +04:00
static void fill_printq_info_52 ( connection_struct * conn , int snum , int uLevel ,
struct pack_desc * desc ,
int count , print_queue_struct * queue ,
print_status_struct * status )
{
2000-07-20 03:47:10 +04:00
int i ;
2000-08-31 11:11:59 +04:00
BOOL ok = False ;
2000-04-19 11:00:22 +04:00
pstring tok , driver , datafile , langmon , helpfile , datatype ;
char * p ;
2000-08-01 00:41:51 +04:00
char * * lines = NULL ;
2000-07-20 00:11:11 +04:00
pstring gen_line ;
2000-08-18 00:05:05 +04:00
BOOL in_tdb = False ;
fstring location ;
2000-12-08 00:55:10 +03:00
pstring drivername ;
2000-07-20 03:47:10 +04:00
/*
* Check in the tdb * first * before checking the legacy
* files . This allows an NT upload to take precedence over
* the existing fileset . JRA .
2000-08-18 00:05:05 +04:00
*
* we need to lookup the driver name prior to making the call
* to get_a_printer_driver_9x_compatible ( ) and not rely on the
* ' print driver ' parameter - - jerry
2000-07-20 03:47:10 +04:00
*/
2000-12-08 00:55:10 +03:00
if ( ( get_driver_name ( snum , drivername ) ) & &
( ( ok = get_a_printer_driver_9x_compatible ( gen_line , drivername ) ) = = True ) )
2000-08-18 00:05:05 +04:00
{
in_tdb = True ;
p = gen_line ;
2000-12-08 00:55:10 +03:00
DEBUG ( 10 , ( " 9x compatable driver line for [%s]: [%s] \n " , drivername , gen_line ) ) ;
2000-08-18 00:05:05 +04:00
}
else
{
2000-08-01 00:41:51 +04:00
/* didn't find driver in tdb */
2000-07-20 03:47:10 +04:00
2000-12-08 00:55:10 +03:00
DEBUG ( 10 , ( " snum: %d \n printerdriver: [%s] \n lp_driverfile: [%s] \n " ,
snum , drivername , lp_driverfile ( snum ) ) ) ;
2000-08-18 00:05:05 +04:00
2001-07-04 11:15:53 +04:00
lines = file_lines_load ( lp_driverfile ( snum ) , NULL ) ;
2000-08-18 00:05:05 +04:00
if ( ! lines )
{
DEBUG ( 3 , ( " Can't open %s - %s \n " , lp_driverfile ( snum ) ,
strerror ( errno ) ) ) ;
desc - > errcode = NERR_notsupported ;
2000-09-01 22:49:26 +04:00
goto done ;
2000-08-18 00:05:05 +04:00
}
2000-08-29 18:38:38 +04:00
/* lookup the long printer driver name in the file description */
for ( i = 0 ; lines [ i ] & & ! ok ; i + + )
2000-08-18 00:05:05 +04:00
{
2000-08-29 18:38:38 +04:00
p = lines [ i ] ;
if ( next_token ( & p , tok , " : " , sizeof ( tok ) ) & &
2000-12-08 00:55:10 +03:00
( strlen ( drivername ) = = strlen ( tok ) ) & &
( ! strncmp ( tok , drivername , strlen ( drivername ) ) ) )
2000-08-18 00:05:05 +04:00
{
2000-08-29 18:38:38 +04:00
ok = True ;
2000-08-18 00:05:05 +04:00
}
2000-07-20 00:11:11 +04:00
}
}
2000-08-01 00:41:51 +04:00
if ( ok )
{
2000-08-18 00:05:05 +04:00
/* driver file name */
if ( ! next_token ( & p , driver , " : " , sizeof ( driver ) ) )
goto err ;
2000-07-20 03:47:10 +04:00
2000-08-18 00:05:05 +04:00
/* data file name */
if ( ! next_token ( & p , datafile , " : " , sizeof ( datafile ) ) )
goto err ;
2000-07-20 03:47:10 +04:00
2000-08-18 00:05:05 +04:00
/*
* for the next tokens - which may be empty - I have
* to check for empty tokens first because the
* next_token function will skip all empty token
* fields */
/* help file */
if ( * p = = ' : ' )
{
* helpfile = ' \0 ' ;
p + + ;
}
else if ( ! next_token ( & p , helpfile , " : " , sizeof ( helpfile ) ) )
goto err ;
2000-04-19 11:00:22 +04:00
2000-08-18 00:05:05 +04:00
/* language monitor */
if ( * p = = ' : ' )
{
* langmon = ' \0 ' ;
p + + ;
}
else if ( ! next_token ( & p , langmon , " : " , sizeof ( langmon ) ) )
goto err ;
2000-04-19 11:00:22 +04:00
2000-08-18 00:05:05 +04:00
/* default data type */
if ( ! next_token ( & p , datatype , " : " , sizeof ( datatype ) ) )
goto err ;
2000-04-19 11:00:22 +04:00
2000-08-18 00:05:05 +04:00
PACKI ( desc , " W " , 0x0400 ) ; /* don't know */
2000-12-08 00:55:10 +03:00
PACKS ( desc , " z " , drivername ) ; /* long printer name */
2000-08-18 00:05:05 +04:00
PACKS ( desc , " z " , driver ) ; /* Driverfile Name */
PACKS ( desc , " z " , datafile ) ; /* Datafile name */
PACKS ( desc , " z " , langmon ) ; /* language monitor */
if ( in_tdb )
{
fstrcpy ( location , " \\ \\ " ) ;
fstrcat ( location , global_myname ) ;
fstrcat ( location , " \\ print$ \\ WIN40 \\ 0 " ) ;
PACKS ( desc , " z " , location ) ; /* share to retrieve files */
}
else
{
PACKS ( desc , " z " , lp_driverlocation ( snum ) ) ; /* share to retrieve files */
}
PACKS ( desc , " z " , datatype ) ; /* default data type */
PACKS ( desc , " z " , helpfile ) ; /* helpfile name */
PACKS ( desc , " z " , driver ) ; /* driver name */
2000-12-08 00:55:10 +03:00
DEBUG ( 3 , ( " printerdriver:%s: \n " , drivername ) ) ;
2000-08-18 00:05:05 +04:00
DEBUG ( 3 , ( " Driver:%s: \n " , driver ) ) ;
DEBUG ( 3 , ( " Data File:%s: \n " , datafile ) ) ;
DEBUG ( 3 , ( " Language Monitor:%s: \n " , langmon ) ) ;
if ( in_tdb )
DEBUG ( 3 , ( " lp_driverlocation:%s: \n " , location ) ) ;
else
DEBUG ( 3 , ( " lp_driverlocation:%s: \n " , lp_driverlocation ( snum ) ) ) ;
DEBUG ( 3 , ( " Data Type:%s: \n " , datatype ) ) ;
DEBUG ( 3 , ( " Help File:%s: \n " , helpfile ) ) ;
PACKI ( desc , " N " , count ) ; /* number of files to copy */
for ( i = 0 ; i < count ; i + + )
{
2000-07-20 03:47:10 +04:00
/* no need to check return value here
* - it was already tested in
* get_printerdrivernumber */
2000-08-18 00:05:05 +04:00
next_token ( & p , tok , " , " , sizeof ( tok ) ) ;
PACKS ( desc , " z " , tok ) ; /* driver files to copy */
DEBUG ( 3 , ( " file:%s: \n " , tok ) ) ;
}
2000-07-20 03:47:10 +04:00
2000-08-18 00:05:05 +04:00
DEBUG ( 3 , ( " fill_printq_info on <%s> gave %d entries \n " ,
SERVICE ( snum ) , count ) ) ;
2000-07-20 03:47:10 +04:00
2000-08-18 00:05:05 +04:00
desc - > errcode = NERR_Success ;
2000-09-01 22:49:26 +04:00
goto done ;
2000-08-01 00:41:51 +04:00
}
2000-07-20 03:47:10 +04:00
err :
DEBUG ( 3 , ( " fill_printq_info: Can't supply driver files \n " ) ) ;
desc - > errcode = NERR_notsupported ;
2000-09-01 22:49:26 +04:00
done :
2000-08-01 00:41:51 +04:00
file_lines_free ( lines ) ;
2000-04-19 11:00:22 +04:00
}
static void fill_printq_info ( connection_struct * conn , int snum , int uLevel ,
struct pack_desc * desc ,
int count , print_queue_struct * queue ,
print_status_struct * status )
{
switch ( uLevel ) {
case 1 :
case 2 :
PACKS ( desc , " B13 " , SERVICE ( snum ) ) ;
break ;
case 3 :
case 4 :
case 5 :
PACKS ( desc , " z " , Expand ( conn , snum , SERVICE ( snum ) ) ) ;
break ;
case 51 :
PACKI ( desc , " K " , printq_status ( status - > status ) ) ;
break ;
}
if ( uLevel = = 1 | | uLevel = = 2 ) {
PACKS ( desc , " B " , " " ) ; /* alignment */
PACKI ( desc , " W " , 5 ) ; /* priority */
PACKI ( desc , " W " , 0 ) ; /* start time */
PACKI ( desc , " W " , 0 ) ; /* until time */
PACKS ( desc , " z " , " " ) ; /* pSepFile */
PACKS ( desc , " z " , " lpd " ) ; /* pPrProc */
PACKS ( desc , " z " , SERVICE ( snum ) ) ; /* pDestinations */
PACKS ( desc , " z " , " " ) ; /* pParms */
if ( snum < 0 ) {
PACKS ( desc , " z " , " UNKNOWN PRINTER " ) ;
PACKI ( desc , " W " , LPSTAT_ERROR ) ;
}
else if ( ! status | | ! status - > message [ 0 ] ) {
PACKS ( desc , " z " , Expand ( conn , snum , lp_comment ( snum ) ) ) ;
PACKI ( desc , " W " , LPSTAT_OK ) ; /* status */
} else {
PACKS ( desc , " z " , status - > message ) ;
PACKI ( desc , " W " , printq_status ( status - > status ) ) ; /* status */
}
PACKI ( desc , ( uLevel = = 1 ? " W " : " N " ) , count ) ;
}
if ( uLevel = = 3 | | uLevel = = 4 ) {
2000-12-08 00:55:10 +03:00
pstring drivername ;
2000-04-19 11:00:22 +04:00
PACKI ( desc , " W " , 5 ) ; /* uPriority */
PACKI ( desc , " W " , 0 ) ; /* uStarttime */
PACKI ( desc , " W " , 0 ) ; /* uUntiltime */
PACKI ( desc , " W " , 5 ) ; /* pad1 */
PACKS ( desc , " z " , " " ) ; /* pszSepFile */
PACKS ( desc , " z " , " WinPrint " ) ; /* pszPrProc */
2001-01-30 00:34:08 +03:00
PACKS ( desc , " z " , NULL ) ; /* pszParms */
2001-01-17 22:16:05 +03:00
PACKS ( desc , " z " , NULL ) ; /* pszComment - don't ask.... JRA */
/* "don't ask" that it's done this way to fix corrupted
Win9X / ME printer comments . */
if ( ! status ) {
2000-04-19 11:00:22 +04:00
PACKI ( desc , " W " , LPSTAT_OK ) ; /* fsStatus */
} else {
PACKI ( desc , " W " , printq_status ( status - > status ) ) ; /* fsStatus */
}
PACKI ( desc , ( uLevel = = 3 ? " W " : " N " ) , count ) ; /* cJobs */
PACKS ( desc , " z " , SERVICE ( snum ) ) ; /* pszPrinters */
2000-12-08 00:55:10 +03:00
get_driver_name ( snum , drivername ) ;
PACKS ( desc , " z " , drivername ) ; /* pszDriverName */
2000-04-19 11:00:22 +04:00
PackDriverData ( desc ) ; /* pDriverData */
}
if ( uLevel = = 2 | | uLevel = = 4 ) {
int i ;
for ( i = 0 ; i < count ; i + + )
fill_printjob_info ( conn , snum , uLevel = = 2 ? 1 : 2 , desc , & queue [ i ] , i ) ;
}
if ( uLevel = = 52 ) {
fill_printq_info_52 ( conn , snum , uLevel , desc , count , queue , status ) ;
}
}
/* This function returns the number of files for a given driver */
static int get_printerdrivernumber ( int snum )
{
2000-09-01 22:49:26 +04:00
int i , result = 0 ;
2000-08-31 11:11:59 +04:00
BOOL ok = False ;
2000-04-19 11:00:22 +04:00
pstring tok ;
char * p ;
2000-08-01 00:41:51 +04:00
char * * lines = NULL ;
2000-07-20 00:11:11 +04:00
pstring gen_line ;
2000-12-08 00:55:10 +03:00
pstring drivername ;
2000-08-18 00:05:05 +04:00
2000-08-01 00:41:51 +04:00
/*
* Check in the tdb * first * before checking the legacy
* files . This allows an NT upload to take precedence over
* the existing fileset . JRA .
2000-08-18 00:05:05 +04:00
*
* we need to lookup the driver name prior to making the call
* to get_a_printer_driver_9x_compatible ( ) and not rely on the
* ' print driver ' parameter - - jerry
2000-08-01 00:41:51 +04:00
*/
2000-12-08 00:55:10 +03:00
if ( ( get_driver_name ( snum , drivername ) ) & &
( ok = get_a_printer_driver_9x_compatible ( gen_line , drivername ) = = True ) )
2000-08-18 00:05:05 +04:00
{
2000-08-01 00:41:51 +04:00
p = gen_line ;
2000-12-08 00:55:10 +03:00
DEBUG ( 10 , ( " 9x compatable driver line for [%s]: [%s] \n " , drivername , gen_line ) ) ;
2000-08-18 00:05:05 +04:00
}
else
{
2000-08-01 00:41:51 +04:00
/* didn't find driver in tdb */
2000-12-08 00:55:10 +03:00
DEBUG ( 10 , ( " snum: %d \n printerdriver: [%s] \n lp_driverfile: [%s] \n " ,
snum , drivername , lp_driverfile ( snum ) ) ) ;
2000-08-01 00:41:51 +04:00
2001-07-04 11:15:53 +04:00
lines = file_lines_load ( lp_driverfile ( snum ) , NULL ) ;
2000-08-18 00:05:05 +04:00
if ( ! lines )
{
2000-08-01 00:41:51 +04:00
DEBUG ( 3 , ( " Can't open %s - %s \n " , lp_driverfile ( snum ) , strerror ( errno ) ) ) ;
2000-09-01 22:49:26 +04:00
goto done ;
2000-08-18 00:05:05 +04:00
}
2000-08-29 18:38:38 +04:00
/* lookup the long printer driver name in the file description */
for ( i = 0 ; lines [ i ] & & ! ok ; i + + )
2000-08-18 00:05:05 +04:00
{
2000-08-29 18:38:38 +04:00
p = lines [ i ] ;
if ( next_token ( & p , tok , " : " , sizeof ( tok ) ) & &
2000-12-08 00:55:10 +03:00
( strlen ( drivername ) = = strlen ( tok ) ) & &
( ! strncmp ( tok , drivername , strlen ( drivername ) ) ) )
2000-08-18 00:05:05 +04:00
{
2000-08-29 18:38:38 +04:00
ok = True ;
2000-08-01 04:41:19 +04:00
}
2000-07-20 00:11:11 +04:00
}
}
2000-04-19 11:00:22 +04:00
2000-08-18 00:05:05 +04:00
if ( ok )
{
2000-04-19 11:00:22 +04:00
/* skip 5 fields */
i = 5 ;
while ( * p & & i ) {
if ( * p + + = = ' : ' ) i - - ;
}
2000-09-01 22:49:26 +04:00
if ( ! * p | | i ) {
DEBUG ( 3 , ( " Can't determine number of printer driver files \n " ) ) ;
goto done ;
}
2000-04-19 11:00:22 +04:00
/* count the number of files */
while ( next_token ( & p , tok , " , " , sizeof ( tok ) ) )
i + + ;
2000-09-01 22:49:26 +04:00
result = i ;
2000-08-01 00:41:51 +04:00
}
2000-09-01 22:49:26 +04:00
done :
2000-08-01 00:41:51 +04:00
file_lines_free ( lines ) ;
2000-09-01 22:49:26 +04:00
return result ;
2000-04-19 11:00:22 +04:00
}
static BOOL api_DosPrintQGetInfo ( connection_struct * conn ,
uint16 vuid , char * param , char * data ,
int mdrcnt , int mprcnt ,
char * * rdata , char * * rparam ,
int * rdata_len , int * rparam_len )
{
2001-08-07 02:39:51 +04:00
char * str1 = param + 2 ;
char * str2 = skip_string ( str1 , 1 ) ;
char * p = skip_string ( str2 , 1 ) ;
char * QueueName = p ;
int uLevel ;
int count = 0 ;
int snum ;
char * str3 ;
struct pack_desc desc ;
print_queue_struct * queue = NULL ;
print_status_struct status ;
char * tmpdata = NULL ;
memset ( ( char * ) & status , ' \0 ' , sizeof ( status ) ) ;
memset ( ( char * ) & desc , ' \0 ' , sizeof ( desc ) ) ;
2000-04-19 11:00:22 +04:00
2001-08-07 02:39:51 +04:00
p = skip_string ( p , 1 ) ;
uLevel = SVAL ( p , 0 ) ;
str3 = p + 4 ;
2000-04-19 11:00:22 +04:00
2001-08-07 02:39:51 +04:00
/* remove any trailing username */
if ( ( p = strchr_m ( QueueName , ' % ' ) ) )
* p = 0 ;
2000-04-19 11:00:22 +04:00
2001-08-07 02:39:51 +04:00
DEBUG ( 3 , ( " api_DosPrintQGetInfo uLevel=%d name=%s \n " , uLevel , QueueName ) ) ;
2000-04-19 11:00:22 +04:00
2001-08-07 02:39:51 +04:00
/* check it's a supported varient */
if ( ! prefix_ok ( str1 , " zWrLh " ) )
return False ;
if ( ! check_printq_info ( & desc , uLevel , str2 , str3 ) ) {
/*
* Patch from Scott Moomaw < scott @ bridgewater . edu >
* to return the ' invalid info level ' error if an
* unknown level was requested .
*/
* rdata_len = 0 ;
* rparam_len = 6 ;
* rparam = REALLOC ( * rparam , * rparam_len ) ;
2001-08-10 09:50:47 +04:00
SSVALS ( * rparam , 0 , ERRunknownlevel ) ;
2001-08-07 02:39:51 +04:00
SSVAL ( * rparam , 2 , 0 ) ;
SSVAL ( * rparam , 4 , 0 ) ;
return ( True ) ;
}
2000-04-19 11:00:22 +04:00
2001-08-07 02:39:51 +04:00
snum = lp_servicenumber ( QueueName ) ;
if ( snum < 0 & & pcap_printername_ok ( QueueName , NULL ) ) {
int pnum = lp_servicenumber ( PRINTERS_NAME ) ;
if ( pnum > = 0 ) {
lp_add_printer ( QueueName , pnum ) ;
snum = lp_servicenumber ( QueueName ) ;
}
}
2000-04-19 11:00:22 +04:00
2001-08-07 02:39:51 +04:00
if ( snum < 0 | | ! VALID_SNUM ( snum ) )
return ( False ) ;
2000-04-19 11:00:22 +04:00
2001-08-07 02:39:51 +04:00
if ( uLevel = = 52 ) {
count = get_printerdrivernumber ( snum ) ;
DEBUG ( 3 , ( " api_DosPrintQGetInfo: Driver files count: %d \n " , count ) ) ;
} else {
count = print_queue_status ( snum , & queue , & status ) ;
}
2000-04-19 11:00:22 +04:00
2001-08-07 02:39:51 +04:00
if ( mdrcnt > 0 ) {
* rdata = REALLOC ( * rdata , mdrcnt ) ;
desc . base = * rdata ;
desc . buflen = mdrcnt ;
} else {
/*
* Don ' t return data but need to get correct length
* init_package will return wrong size if buflen = 0
*/
desc . buflen = getlen ( desc . format ) ;
desc . base = tmpdata = ( char * ) malloc ( desc . buflen ) ;
}
2001-03-23 06:12:58 +03:00
2001-08-07 02:39:51 +04:00
if ( init_package ( & desc , 1 , count ) ) {
desc . subcount = count ;
fill_printq_info ( conn , snum , uLevel , & desc , count , queue , & status ) ;
}
2000-04-19 11:00:22 +04:00
2001-08-07 02:39:51 +04:00
* rdata_len = desc . usedlen ;
2000-04-19 11:00:22 +04:00
2001-08-07 02:39:51 +04:00
/*
* We must set the return code to ERRbuftoosmall
* in order to support lanman style printing with Win NT / 2 k
* clients - - jerry
*/
2001-08-10 08:27:26 +04:00
if ( ! mdrcnt & & lp_disable_spoolss ( ) )
2001-08-07 02:39:51 +04:00
desc . errcode = ERRbuftoosmall ;
* rdata_len = desc . usedlen ;
* rparam_len = 6 ;
* rparam = REALLOC ( * rparam , * rparam_len ) ;
SSVALS ( * rparam , 0 , desc . errcode ) ;
SSVAL ( * rparam , 2 , 0 ) ;
SSVAL ( * rparam , 4 , desc . neededlen ) ;
2000-04-19 11:00:22 +04:00
2001-08-07 02:39:51 +04:00
DEBUG ( 4 , ( " printqgetinfo: errorcode %d \n " , desc . errcode ) ) ;
2000-04-19 11:00:22 +04:00
2001-09-17 15:25:41 +04:00
SAFE_FREE ( queue ) ;
SAFE_FREE ( tmpdata ) ;
2001-03-23 06:12:58 +03:00
2001-08-07 02:39:51 +04:00
return ( True ) ;
2000-04-19 11:00:22 +04:00
}
/****************************************************************************
2000-08-01 04:41:19 +04:00
View list of all print jobs on all queues .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2000-04-19 11:00:22 +04:00
static BOOL api_DosPrintQEnum ( connection_struct * conn , uint16 vuid , char * param , char * data ,
int mdrcnt , int mprcnt ,
char * * rdata , char * * rparam ,
int * rdata_len , int * rparam_len )
{
char * param_format = param + 2 ;
char * output_format1 = skip_string ( param_format , 1 ) ;
char * p = skip_string ( output_format1 , 1 ) ;
int uLevel = SVAL ( p , 0 ) ;
char * output_format2 = p + 4 ;
int services = lp_numservices ( ) ;
int i , n ;
struct pack_desc desc ;
print_queue_struct * * queue = NULL ;
print_status_struct * status = NULL ;
int * subcntarr = NULL ;
int queuecnt , subcnt = 0 , succnt = 0 ;
memset ( ( char * ) & desc , ' \0 ' , sizeof ( desc ) ) ;
DEBUG ( 3 , ( " DosPrintQEnum uLevel=%d \n " , uLevel ) ) ;
if ( ! prefix_ok ( param_format , " WrLeh " ) ) return False ;
if ( ! check_printq_info ( & desc , uLevel , output_format1 , output_format2 ) ) {
/*
* Patch from Scott Moomaw < scott @ bridgewater . edu >
* to return the ' invalid info level ' error if an
* unknown level was requested .
*/
* rdata_len = 0 ;
* rparam_len = 6 ;
* rparam = REALLOC ( * rparam , * rparam_len ) ;
2001-08-10 09:50:47 +04:00
SSVALS ( * rparam , 0 , ERRunknownlevel ) ;
2000-04-19 11:00:22 +04:00
SSVAL ( * rparam , 2 , 0 ) ;
SSVAL ( * rparam , 4 , 0 ) ;
return ( True ) ;
}
queuecnt = 0 ;
for ( i = 0 ; i < services ; i + + )
if ( lp_snum_ok ( i ) & & lp_print_ok ( i ) & & lp_browseable ( i ) )
queuecnt + + ;
if ( uLevel > 0 ) {
if ( ( queue = ( print_queue_struct * * ) malloc ( queuecnt * sizeof ( print_queue_struct * ) ) ) = = NULL ) {
DEBUG ( 0 , ( " api_DosPrintQEnum: malloc fail ! \n " ) ) ;
return False ;
}
memset ( queue , 0 , queuecnt * sizeof ( print_queue_struct * ) ) ;
if ( ( status = ( print_status_struct * ) malloc ( queuecnt * sizeof ( print_status_struct ) ) ) = = NULL ) {
DEBUG ( 0 , ( " api_DosPrintQEnum: malloc fail ! \n " ) ) ;
return False ;
}
memset ( status , 0 , queuecnt * sizeof ( print_status_struct ) ) ;
if ( ( subcntarr = ( int * ) malloc ( queuecnt * sizeof ( int ) ) ) = = NULL ) {
DEBUG ( 0 , ( " api_DosPrintQEnum: malloc fail ! \n " ) ) ;
return False ;
}
subcnt = 0 ;
n = 0 ;
for ( i = 0 ; i < services ; i + + )
if ( lp_snum_ok ( i ) & & lp_print_ok ( i ) & & lp_browseable ( i ) ) {
subcntarr [ n ] = print_queue_status ( i , & queue [ n ] , & status [ n ] ) ;
subcnt + = subcntarr [ n ] ;
n + + ;
}
}
if ( mdrcnt > 0 ) * rdata = REALLOC ( * rdata , mdrcnt ) ;
desc . base = * rdata ;
desc . buflen = mdrcnt ;
if ( init_package ( & desc , queuecnt , subcnt ) ) {
n = 0 ;
succnt = 0 ;
for ( i = 0 ; i < services ; i + + )
if ( lp_snum_ok ( i ) & & lp_print_ok ( i ) & & lp_browseable ( i ) ) {
fill_printq_info ( conn , i , uLevel , & desc , subcntarr [ n ] , queue [ n ] , & status [ n ] ) ;
n + + ;
if ( desc . errcode = = NERR_Success ) succnt = n ;
}
}
2001-09-17 15:25:41 +04:00
SAFE_FREE ( subcntarr ) ;
2000-04-19 11:00:22 +04:00
* rdata_len = desc . usedlen ;
* rparam_len = 8 ;
* rparam = REALLOC ( * rparam , * rparam_len ) ;
SSVALS ( * rparam , 0 , desc . errcode ) ;
SSVAL ( * rparam , 2 , 0 ) ;
SSVAL ( * rparam , 4 , succnt ) ;
SSVAL ( * rparam , 6 , queuecnt ) ;
for ( i = 0 ; i < queuecnt ; i + + ) {
2001-09-17 15:25:41 +04:00
if ( queue ) SAFE_FREE ( queue [ i ] ) ;
2000-04-19 11:00:22 +04:00
}
2001-09-17 15:25:41 +04:00
SAFE_FREE ( queue ) ;
SAFE_FREE ( status ) ;
2000-04-19 11:00:22 +04:00
return True ;
}
/****************************************************************************
get info level for a server list query
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static BOOL check_server_info ( int uLevel , char * id )
{
switch ( uLevel ) {
case 0 :
if ( strcmp ( id , " B16 " ) ! = 0 ) return False ;
break ;
case 1 :
if ( strcmp ( id , " B16BBDz " ) ! = 0 ) return False ;
break ;
default :
return False ;
}
return True ;
}
struct srv_info_struct
{
fstring name ;
uint32 type ;
fstring comment ;
fstring domain ;
BOOL server_added ;
} ;
/*******************************************************************
get server info lists from the files saved by nmbd . Return the
number of entries
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static int get_server_info ( uint32 servertype ,
struct srv_info_struct * * servers ,
char * domain )
{
int count = 0 ;
int alloced = 0 ;
char * * lines ;
BOOL local_list_only ;
int i ;
2001-07-04 11:15:53 +04:00
lines = file_lines_load ( lock_path ( SERVER_LIST ) , NULL ) ;
2000-04-19 11:00:22 +04:00
if ( ! lines ) {
DEBUG ( 4 , ( " Can't open %s - %s \n " , lock_path ( SERVER_LIST ) , strerror ( errno ) ) ) ;
return ( 0 ) ;
}
/* request for everything is code for request all servers */
if ( servertype = = SV_TYPE_ALL )
servertype & = ~ ( SV_TYPE_DOMAIN_ENUM | SV_TYPE_LOCAL_LIST_ONLY ) ;
local_list_only = ( servertype & SV_TYPE_LOCAL_LIST_ONLY ) ;
DEBUG ( 4 , ( " Servertype search: %8x \n " , servertype ) ) ;
for ( i = 0 ; lines [ i ] ; i + + ) {
fstring stype ;
struct srv_info_struct * s ;
char * ptr = lines [ i ] ;
BOOL ok = True ;
if ( ! * ptr ) continue ;
if ( count = = alloced ) {
2001-08-12 21:30:01 +04:00
struct srv_info_struct * ts ;
2000-04-19 11:00:22 +04:00
alloced + = 10 ;
2001-08-12 21:30:01 +04:00
ts = ( struct srv_info_struct * )
2000-04-19 11:00:22 +04:00
Realloc ( * servers , sizeof ( * * servers ) * alloced ) ;
2001-08-12 21:30:01 +04:00
if ( ! ts ) {
DEBUG ( 0 , ( " get_server_info: failed to enlarge servers info struct! \n " ) ) ;
return ( 0 ) ;
}
else * servers = ts ;
2000-04-19 11:00:22 +04:00
memset ( ( char * ) ( ( * servers ) + count ) , ' \0 ' , sizeof ( * * servers ) * ( alloced - count ) ) ;
}
s = & ( * servers ) [ count ] ;
if ( ! next_token ( & ptr , s - > name , NULL , sizeof ( s - > name ) ) ) continue ;
if ( ! next_token ( & ptr , stype , NULL , sizeof ( stype ) ) ) continue ;
if ( ! next_token ( & ptr , s - > comment , NULL , sizeof ( s - > comment ) ) ) continue ;
if ( ! next_token ( & ptr , s - > domain , NULL , sizeof ( s - > domain ) ) ) {
/* this allows us to cope with an old nmbd */
pstrcpy ( s - > domain , global_myworkgroup ) ;
}
if ( sscanf ( stype , " %X " , & s - > type ) ! = 1 ) {
DEBUG ( 4 , ( " r:host file " ) ) ;
ok = False ;
}
/* Filter the servers/domains we return based on what was asked for. */
/* Check to see if we are being asked for a local list only. */
if ( local_list_only & & ( ( s - > type & SV_TYPE_LOCAL_LIST_ONLY ) = = 0 ) ) {
DEBUG ( 4 , ( " r: local list only " ) ) ;
ok = False ;
}
/* doesn't match up: don't want it */
if ( ! ( servertype & s - > type ) ) {
DEBUG ( 4 , ( " r:serv type " ) ) ;
ok = False ;
}
if ( ( servertype & SV_TYPE_DOMAIN_ENUM ) ! =
( s - > type & SV_TYPE_DOMAIN_ENUM ) )
{
DEBUG ( 4 , ( " s: dom mismatch " ) ) ;
ok = False ;
}
if ( ! strequal ( domain , s - > domain ) & & ! ( servertype & SV_TYPE_DOMAIN_ENUM ) )
{
ok = False ;
}
/* We should never return a server type with a SV_TYPE_LOCAL_LIST_ONLY set. */
s - > type & = ~ SV_TYPE_LOCAL_LIST_ONLY ;
if ( ok )
{
DEBUG ( 4 , ( " **SV** %20s %8x %25s %15s \n " ,
s - > name , s - > type , s - > comment , s - > domain ) ) ;
s - > server_added = True ;
count + + ;
}
else
{
DEBUG ( 4 , ( " %20s %8x %25s %15s \n " ,
s - > name , s - > type , s - > comment , s - > domain ) ) ;
}
}
file_lines_free ( lines ) ;
return ( count ) ;
}
/*******************************************************************
fill in a server info structure
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static int fill_srv_info ( struct srv_info_struct * service ,
int uLevel , char * * buf , int * buflen ,
char * * stringbuf , int * stringspace , char * baseaddr )
{
int struct_len ;
char * p ;
char * p2 ;
int l2 ;
int len ;
switch ( uLevel ) {
case 0 : struct_len = 16 ; break ;
case 1 : struct_len = 26 ; break ;
default : return - 1 ;
}
if ( ! buf )
{
len = 0 ;
switch ( uLevel )
{
case 1 :
len = strlen ( service - > comment ) + 1 ;
break ;
}
if ( buflen ) * buflen = struct_len ;
if ( stringspace ) * stringspace = len ;
return struct_len + len ;
}
len = struct_len ;
p = * buf ;
if ( * buflen < struct_len ) return - 1 ;
if ( stringbuf )
{
p2 = * stringbuf ;
l2 = * stringspace ;
}
else
{
p2 = p + struct_len ;
l2 = * buflen - struct_len ;
}
if ( ! baseaddr ) baseaddr = p ;
switch ( uLevel )
{
case 0 :
2001-07-04 11:15:53 +04:00
push_ascii ( p , service - > name , 15 , STR_TERMINATE ) ;
break ;
2000-04-19 11:00:22 +04:00
case 1 :
2001-07-04 11:15:53 +04:00
push_ascii ( p , service - > name , 15 , STR_TERMINATE ) ;
SIVAL ( p , 18 , service - > type ) ;
SIVAL ( p , 22 , PTR_DIFF ( p2 , baseaddr ) ) ;
len + = CopyAndAdvance ( & p2 , service - > comment , & l2 ) ;
break ;
2000-04-19 11:00:22 +04:00
}
if ( stringbuf )
{
* buf = p + struct_len ;
* buflen - = struct_len ;
* stringbuf = p2 ;
* stringspace = l2 ;
}
else
{
* buf = p2 ;
* buflen - = len ;
}
return len ;
}
static BOOL srv_comp ( struct srv_info_struct * s1 , struct srv_info_struct * s2 )
{
return ( strcmp ( s1 - > name , s2 - > name ) ) ;
}
/****************************************************************************
view list of servers available ( or possibly domains ) . The info is
extracted from lists saved by nmbd on the local host
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static BOOL api_RNetServerEnum ( connection_struct * conn , uint16 vuid , char * param , char * data ,
int mdrcnt , int mprcnt , char * * rdata ,
char * * rparam , int * rdata_len , int * rparam_len )
{
char * str1 = param + 2 ;
char * str2 = skip_string ( str1 , 1 ) ;
char * p = skip_string ( str2 , 1 ) ;
int uLevel = SVAL ( p , 0 ) ;
int buf_len = SVAL ( p , 2 ) ;
uint32 servertype = IVAL ( p , 4 ) ;
char * p2 ;
int data_len , fixed_len , string_len ;
int f_len = 0 , s_len = 0 ;
struct srv_info_struct * servers = NULL ;
int counted = 0 , total = 0 ;
int i , missed ;
fstring domain ;
BOOL domain_request ;
BOOL local_request ;
/* If someone sets all the bits they don't really mean to set
DOMAIN_ENUM and LOCAL_LIST_ONLY , they just want all the
known servers . */
if ( servertype = = SV_TYPE_ALL )
servertype & = ~ ( SV_TYPE_DOMAIN_ENUM | SV_TYPE_LOCAL_LIST_ONLY ) ;
/* If someone sets SV_TYPE_LOCAL_LIST_ONLY but hasn't set
any other bit ( they may just set this bit on it ' s own ) they
want all the locally seen servers . However this bit can be
set on its own so set the requested servers to be
ALL - DOMAIN_ENUM . */
if ( ( servertype & SV_TYPE_LOCAL_LIST_ONLY ) & & ! ( servertype & SV_TYPE_DOMAIN_ENUM ) )
servertype = SV_TYPE_ALL & ~ ( SV_TYPE_DOMAIN_ENUM ) ;
domain_request = ( ( servertype & SV_TYPE_DOMAIN_ENUM ) ! = 0 ) ;
local_request = ( ( servertype & SV_TYPE_LOCAL_LIST_ONLY ) ! = 0 ) ;
p + = 8 ;
if ( ! prefix_ok ( str1 , " WrLehD " ) ) return False ;
if ( ! check_server_info ( uLevel , str2 ) ) return False ;
DEBUG ( 4 , ( " server request level: %s %8x " , str2 , servertype ) ) ;
DEBUG ( 4 , ( " domains_req:%s " , BOOLSTR ( domain_request ) ) ) ;
DEBUG ( 4 , ( " local_only:%s \n " , BOOLSTR ( local_request ) ) ) ;
if ( strcmp ( str1 , " WrLehDz " ) = = 0 ) {
2001-07-04 11:15:53 +04:00
pull_ascii_fstring ( domain , p ) ;
2000-04-19 11:00:22 +04:00
} else {
2001-03-31 17:48:24 +04:00
fstrcpy ( domain , global_myworkgroup ) ;
2000-04-19 11:00:22 +04:00
}
if ( lp_browse_list ( ) )
total = get_server_info ( servertype , & servers , domain ) ;
data_len = fixed_len = string_len = 0 ;
missed = 0 ;
2001-01-24 22:04:56 +03:00
if ( total > 0 )
qsort ( servers , total , sizeof ( servers [ 0 ] ) , QSORT_CAST srv_comp ) ;
2000-04-19 11:00:22 +04:00
{
char * lastname = NULL ;
for ( i = 0 ; i < total ; i + + )
{
struct srv_info_struct * s = & servers [ i ] ;
if ( lastname & & strequal ( lastname , s - > name ) ) continue ;
lastname = s - > name ;
data_len + = fill_srv_info ( s , uLevel , 0 , & f_len , 0 , & s_len , 0 ) ;
DEBUG ( 4 , ( " fill_srv_info %20s %8x %25s %15s \n " ,
s - > name , s - > type , s - > comment , s - > domain ) ) ;
if ( data_len < = buf_len ) {
counted + + ;
fixed_len + = f_len ;
string_len + = s_len ;
} else {
missed + + ;
}
}
}
* rdata_len = fixed_len + string_len ;
* rdata = REALLOC ( * rdata , * rdata_len ) ;
memset ( * rdata , ' \0 ' , * rdata_len ) ;
p2 = ( * rdata ) + fixed_len ; /* auxilliary data (strings) will go here */
p = * rdata ;
f_len = fixed_len ;
s_len = string_len ;
{
char * lastname = NULL ;
int count2 = counted ;
for ( i = 0 ; i < total & & count2 ; i + + )
{
struct srv_info_struct * s = & servers [ i ] ;
if ( lastname & & strequal ( lastname , s - > name ) ) continue ;
lastname = s - > name ;
fill_srv_info ( s , uLevel , & p , & f_len , & p2 , & s_len , * rdata ) ;
DEBUG ( 4 , ( " fill_srv_info %20s %8x %25s %15s \n " ,
s - > name , s - > type , s - > comment , s - > domain ) ) ;
count2 - - ;
}
}
* rparam_len = 8 ;
* rparam = REALLOC ( * rparam , * rparam_len ) ;
SSVAL ( * rparam , 0 , ( missed = = 0 ? NERR_Success : ERRmoredata ) ) ;
SSVAL ( * rparam , 2 , 0 ) ;
SSVAL ( * rparam , 4 , counted ) ;
SSVAL ( * rparam , 6 , counted + missed ) ;
2001-09-17 15:25:41 +04:00
SAFE_FREE ( servers ) ;
2000-04-19 11:00:22 +04:00
DEBUG ( 3 , ( " NetServerEnum domain = %s uLevel=%d counted=%d total=%d \n " ,
domain , uLevel , counted , counted + missed ) ) ;
return ( True ) ;
}
/****************************************************************************
command 0x34 - suspected of being a " Lookup Names " stub api
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static BOOL api_RNetGroupGetUsers ( connection_struct * conn , uint16 vuid , char * param , char * data ,
int mdrcnt , int mprcnt , char * * rdata ,
char * * rparam , int * rdata_len , int * rparam_len )
{
char * str1 = param + 2 ;
char * str2 = skip_string ( str1 , 1 ) ;
char * p = skip_string ( str2 , 1 ) ;
int uLevel = SVAL ( p , 0 ) ;
int buf_len = SVAL ( p , 2 ) ;
int counted = 0 ;
int missed = 0 ;
DEBUG ( 5 , ( " RNetGroupGetUsers: %s %s %s %d %d \n " ,
str1 , str2 , p , uLevel , buf_len ) ) ;
if ( ! prefix_ok ( str1 , " zWrLeh " ) ) return False ;
* rdata_len = 0 ;
* rparam_len = 8 ;
* rparam = REALLOC ( * rparam , * rparam_len ) ;
SSVAL ( * rparam , 0 , 0x08AC ) ; /* informational warning message */
SSVAL ( * rparam , 2 , 0 ) ;
SSVAL ( * rparam , 4 , counted ) ;
SSVAL ( * rparam , 6 , counted + missed ) ;
return ( True ) ;
}
/****************************************************************************
get info about a share
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static BOOL check_share_info ( int uLevel , char * id )
{
switch ( uLevel ) {
case 0 :
if ( strcmp ( id , " B13 " ) ! = 0 ) return False ;
break ;
case 1 :
if ( strcmp ( id , " B13BWz " ) ! = 0 ) return False ;
break ;
case 2 :
if ( strcmp ( id , " B13BWzWWWzB9B " ) ! = 0 ) return False ;
break ;
case 91 :
if ( strcmp ( id , " B13BWzWWWzB9BB9BWzWWzWW " ) ! = 0 ) return False ;
break ;
default : return False ;
}
return True ;
}
static int fill_share_info ( connection_struct * conn , int snum , int uLevel ,
char * * buf , int * buflen ,
char * * stringbuf , int * stringspace , char * baseaddr )
{
int struct_len ;
char * p ;
char * p2 ;
int l2 ;
int len ;
switch ( uLevel ) {
case 0 : struct_len = 13 ; break ;
case 1 : struct_len = 20 ; break ;
case 2 : struct_len = 40 ; break ;
case 91 : struct_len = 68 ; break ;
default : return - 1 ;
}
if ( ! buf )
{
len = 0 ;
if ( uLevel > 0 ) len + = StrlenExpanded ( conn , snum , lp_comment ( snum ) ) ;
if ( uLevel > 1 ) len + = strlen ( lp_pathname ( snum ) ) + 1 ;
if ( buflen ) * buflen = struct_len ;
if ( stringspace ) * stringspace = len ;
return struct_len + len ;
}
len = struct_len ;
p = * buf ;
if ( ( * buflen ) < struct_len ) return - 1 ;
if ( stringbuf )
{
p2 = * stringbuf ;
l2 = * stringspace ;
}
else
{
p2 = p + struct_len ;
l2 = ( * buflen ) - struct_len ;
}
if ( ! baseaddr ) baseaddr = p ;
2001-07-04 11:15:53 +04:00
push_ascii ( p , lp_servicename ( snum ) , 13 , STR_TERMINATE ) ;
2000-04-19 11:00:22 +04:00
if ( uLevel > 0 )
{
int type ;
CVAL ( p , 13 ) = 0 ;
type = STYPE_DISKTREE ;
if ( lp_print_ok ( snum ) ) type = STYPE_PRINTQ ;
2001-03-17 17:02:44 +03:00
if ( strequal ( " IPC " , lp_fstype ( snum ) ) ) type = STYPE_IPC ;
2000-04-19 11:00:22 +04:00
SSVAL ( p , 14 , type ) ; /* device type */
SIVAL ( p , 16 , PTR_DIFF ( p2 , baseaddr ) ) ;
len + = CopyExpanded ( conn , snum , & p2 , lp_comment ( snum ) , & l2 ) ;
}
if ( uLevel > 1 )
{
SSVAL ( p , 20 , ACCESS_READ | ACCESS_WRITE | ACCESS_CREATE ) ; /* permissions */
SSVALS ( p , 22 , - 1 ) ; /* max uses */
SSVAL ( p , 24 , 1 ) ; /* current uses */
SIVAL ( p , 26 , PTR_DIFF ( p2 , baseaddr ) ) ; /* local pathname */
len + = CopyAndAdvance ( & p2 , lp_pathname ( snum ) , & l2 ) ;
memset ( p + 30 , 0 , SHPWLEN + 2 ) ; /* passwd (reserved), pad field */
}
if ( uLevel > 2 )
{
memset ( p + 40 , 0 , SHPWLEN + 2 ) ;
SSVAL ( p , 50 , 0 ) ;
SIVAL ( p , 52 , 0 ) ;
SSVAL ( p , 56 , 0 ) ;
SSVAL ( p , 58 , 0 ) ;
SIVAL ( p , 60 , 0 ) ;
SSVAL ( p , 64 , 0 ) ;
SSVAL ( p , 66 , 0 ) ;
}
if ( stringbuf )
{
( * buf ) = p + struct_len ;
( * buflen ) - = struct_len ;
( * stringbuf ) = p2 ;
( * stringspace ) = l2 ;
}
else
{
( * buf ) = p2 ;
( * buflen ) - = len ;
}
return len ;
}
static BOOL api_RNetShareGetInfo ( connection_struct * conn , uint16 vuid , char * param , char * data ,
int mdrcnt , int mprcnt ,
char * * rdata , char * * rparam ,
int * rdata_len , int * rparam_len )
{
char * str1 = param + 2 ;
char * str2 = skip_string ( str1 , 1 ) ;
char * netname = skip_string ( str2 , 1 ) ;
char * p = skip_string ( netname , 1 ) ;
int uLevel = SVAL ( p , 0 ) ;
int snum = find_service ( netname ) ;
if ( snum < 0 ) return False ;
/* check it's a supported varient */
if ( ! prefix_ok ( str1 , " zWrLh " ) ) return False ;
if ( ! check_share_info ( uLevel , str2 ) ) return False ;
* rdata = REALLOC ( * rdata , mdrcnt ) ;
p = * rdata ;
* rdata_len = fill_share_info ( conn , snum , uLevel , & p , & mdrcnt , 0 , 0 , 0 ) ;
if ( * rdata_len < 0 ) return False ;
* rparam_len = 6 ;
* rparam = REALLOC ( * rparam , * rparam_len ) ;
SSVAL ( * rparam , 0 , NERR_Success ) ;
SSVAL ( * rparam , 2 , 0 ) ; /* converter word */
SSVAL ( * rparam , 4 , * rdata_len ) ;
return ( True ) ;
}
/****************************************************************************
view list of shares available
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static BOOL api_RNetShareEnum ( connection_struct * conn , uint16 vuid , char * param , char * data ,
int mdrcnt , int mprcnt ,
char * * rdata , char * * rparam ,
int * rdata_len , int * rparam_len )
{
char * str1 = param + 2 ;
char * str2 = skip_string ( str1 , 1 ) ;
char * p = skip_string ( str2 , 1 ) ;
int uLevel = SVAL ( p , 0 ) ;
int buf_len = SVAL ( p , 2 ) ;
char * p2 ;
int count = lp_numservices ( ) ;
int total = 0 , counted = 0 ;
BOOL missed = False ;
int i ;
int data_len , fixed_len , string_len ;
int f_len = 0 , s_len = 0 ;
if ( ! prefix_ok ( str1 , " WrLeh " ) ) return False ;
if ( ! check_share_info ( uLevel , str2 ) ) return False ;
data_len = fixed_len = string_len = 0 ;
for ( i = 0 ; i < count ; i + + )
if ( lp_browseable ( i ) & & lp_snum_ok ( i ) )
{
total + + ;
data_len + = fill_share_info ( conn , i , uLevel , 0 , & f_len , 0 , & s_len , 0 ) ;
if ( data_len < = buf_len )
{
counted + + ;
fixed_len + = f_len ;
string_len + = s_len ;
}
else
missed = True ;
}
* rdata_len = fixed_len + string_len ;
* rdata = REALLOC ( * rdata , * rdata_len ) ;
memset ( * rdata , 0 , * rdata_len ) ;
p2 = ( * rdata ) + fixed_len ; /* auxillery data (strings) will go here */
p = * rdata ;
f_len = fixed_len ;
s_len = string_len ;
for ( i = 0 ; i < count ; i + + )
if ( lp_browseable ( i ) & & lp_snum_ok ( i ) )
if ( fill_share_info ( conn , i , uLevel , & p , & f_len , & p2 , & s_len , * rdata ) < 0 )
break ;
* rparam_len = 8 ;
* rparam = REALLOC ( * rparam , * rparam_len ) ;
SSVAL ( * rparam , 0 , missed ? ERRmoredata : NERR_Success ) ;
SSVAL ( * rparam , 2 , 0 ) ;
SSVAL ( * rparam , 4 , counted ) ;
SSVAL ( * rparam , 6 , total ) ;
DEBUG ( 3 , ( " RNetShareEnum gave %d entries of %d (%d %d %d %d) \n " ,
counted , total , uLevel ,
buf_len , * rdata_len , mdrcnt ) ) ;
return ( True ) ;
}
2001-08-28 20:05:55 +04:00
/****************************************************************************
view list of groups available
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static BOOL api_RNetGroupEnum ( connection_struct * conn , uint16 vuid , char * param , char * data ,
int mdrcnt , int mprcnt ,
char * * rdata , char * * rparam ,
int * rdata_len , int * rparam_len )
{
char * str1 = param + 2 ;
char * str2 = skip_string ( str1 , 1 ) ;
char * p = skip_string ( str2 , 1 ) ;
int uLevel = SVAL ( p , 0 ) ;
char * p2 ;
int count = 0 ;
if ( ! prefix_ok ( str1 , " WrLeh " ) ) return False ;
/* check it's a supported variant */
switch ( uLevel )
{
case 0 :
p2 = " B21 " ;
break ;
default :
return False ;
}
if ( strcmp ( p2 , str2 ) ! = 0 ) return False ;
* rdata_len = mdrcnt + 1024 ;
* rdata = REALLOC ( * rdata , * rdata_len ) ;
SSVAL ( * rparam , 0 , NERR_Success ) ;
SSVAL ( * rparam , 2 , 0 ) ; /* converter word */
p = * rdata ;
/* XXXX we need a real SAM database some day */
pstrcpy ( p , " Users " ) ; p + = 21 ; count + + ;
pstrcpy ( p , " Domain Users " ) ; p + = 21 ; count + + ;
pstrcpy ( p , " Guests " ) ; p + = 21 ; count + + ;
pstrcpy ( p , " Domain Guests " ) ; p + = 21 ; count + + ;
* rdata_len = PTR_DIFF ( p , * rdata ) ;
* rparam_len = 8 ;
* rparam = REALLOC ( * rparam , * rparam_len ) ;
SSVAL ( * rparam , 4 , count ) ; /* is this right?? */
SSVAL ( * rparam , 6 , count ) ; /* is this right?? */
DEBUG ( 3 , ( " api_RNetGroupEnum gave %d entries \n " , count ) ) ;
return ( True ) ;
}
/****************************************************************************
view list of groups available
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static BOOL api_RNetUserEnum ( connection_struct * conn , uint16 vuid , char * param , char * data ,
int mdrcnt , int mprcnt ,
char * * rdata , char * * rparam ,
int * rdata_len , int * rparam_len )
{
char * str1 = param + 2 ;
char * str2 = skip_string ( str1 , 1 ) ;
char * p = skip_string ( str2 , 1 ) ;
int uLevel = SVAL ( p , 0 ) ;
char * p2 ;
int count = 0 ;
if ( ! prefix_ok ( str1 , " WrLeh " ) ) return False ;
/* check it's a supported variant */
switch ( uLevel )
{
case 0 :
p2 = " B21 " ;
break ;
default :
return False ;
}
if ( strcmp ( p2 , str2 ) ! = 0 ) return False ;
* rdata_len = mdrcnt + 1024 ;
* rdata = REALLOC ( * rdata , * rdata_len ) ;
SSVAL ( * rparam , 0 , NERR_Success ) ;
SSVAL ( * rparam , 2 , 0 ) ; /* converter word */
p = * rdata ;
/* XXXX we need a real SAM database some day */
pstrcpy ( p , " Users " ) ; p + = 21 ; count + + ;
pstrcpy ( p , " Domain Users " ) ; p + = 21 ; count + + ;
pstrcpy ( p , " Guests " ) ; p + = 21 ; count + + ;
pstrcpy ( p , " Domain Guests " ) ; p + = 21 ; count + + ;
* rdata_len = PTR_DIFF ( p , * rdata ) ;
* rparam_len = 8 ;
* rparam = REALLOC ( * rparam , * rparam_len ) ;
SSVAL ( * rparam , 4 , count ) ; /* is this right?? */
SSVAL ( * rparam , 6 , count ) ; /* is this right?? */
DEBUG ( 3 , ( " api_RNetUserEnum gave %d entries \n " , count ) ) ;
return ( True ) ;
}
2000-04-19 11:00:22 +04:00
/****************************************************************************
get the time of day info
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static BOOL api_NetRemoteTOD ( connection_struct * conn , uint16 vuid , char * param , char * data ,
int mdrcnt , int mprcnt ,
char * * rdata , char * * rparam ,
int * rdata_len , int * rparam_len )
{
char * p ;
* rparam_len = 4 ;
* rparam = REALLOC ( * rparam , * rparam_len ) ;
* rdata_len = 21 ;
* rdata = REALLOC ( * rdata , * rdata_len ) ;
SSVAL ( * rparam , 0 , NERR_Success ) ;
SSVAL ( * rparam , 2 , 0 ) ; /* converter word */
p = * rdata ;
{
struct tm * t ;
time_t unixdate = time ( NULL ) ;
put_dos_date3 ( p , 0 , unixdate ) ; /* this is the time that is looked at
by NT in a " net time " operation ,
it seems to ignore the one below */
/* the client expects to get localtime, not GMT, in this bit
( I think , this needs testing ) */
t = LocalTime ( & unixdate ) ;
SIVAL ( p , 4 , 0 ) ; /* msecs ? */
CVAL ( p , 8 ) = t - > tm_hour ;
CVAL ( p , 9 ) = t - > tm_min ;
CVAL ( p , 10 ) = t - > tm_sec ;
CVAL ( p , 11 ) = 0 ; /* hundredths of seconds */
SSVALS ( p , 12 , TimeDiff ( unixdate ) / 60 ) ; /* timezone in minutes from GMT */
SSVAL ( p , 14 , 10000 ) ; /* timer interval in 0.0001 of sec */
CVAL ( p , 16 ) = t - > tm_mday ;
CVAL ( p , 17 ) = t - > tm_mon + 1 ;
SSVAL ( p , 18 , 1900 + t - > tm_year ) ;
CVAL ( p , 20 ) = t - > tm_wday ;
}
return ( True ) ;
}
/****************************************************************************
Set the user password .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static BOOL api_SetUserPassword ( connection_struct * conn , uint16 vuid , char * param , char * data ,
int mdrcnt , int mprcnt ,
char * * rdata , char * * rparam ,
int * rdata_len , int * rparam_len )
{
char * p = skip_string ( param + 2 , 2 ) ;
fstring user ;
fstring pass1 , pass2 ;
2001-07-04 11:15:53 +04:00
pull_ascii_fstring ( user , p ) ;
2000-04-19 11:00:22 +04:00
p = skip_string ( p , 1 ) ;
memset ( pass1 , ' \0 ' , sizeof ( pass1 ) ) ;
memset ( pass2 , ' \0 ' , sizeof ( pass2 ) ) ;
memcpy ( pass1 , p , 16 ) ;
memcpy ( pass2 , p + 16 , 16 ) ;
* rparam_len = 4 ;
* rparam = REALLOC ( * rparam , * rparam_len ) ;
* rdata_len = 0 ;
SSVAL ( * rparam , 0 , NERR_badpass ) ;
SSVAL ( * rparam , 2 , 0 ) ; /* converter word */
DEBUG ( 3 , ( " Set password for <%s> \n " , user ) ) ;
/*
* Pass the user through the NT - > unix user mapping
* function .
*/
( void ) map_username ( user ) ;
/*
* Do any UNIX username case mangling .
*/
( void ) Get_Pwnam ( user , True ) ;
/*
* Attempt to verify the old password against smbpasswd entries
* Win98 clients send old and new password in plaintext for this call .
*/
{
fstring saved_pass2 ;
2001-05-04 19:44:27 +04:00
SAM_ACCOUNT * sampass = NULL ;
2000-04-19 11:00:22 +04:00
/*
* Save the new password as change_oem_password overwrites it
* with zeros .
*/
fstrcpy ( saved_pass2 , pass2 ) ;
2000-11-14 02:03:34 +03:00
if ( check_plaintext_password ( user , pass1 , strlen ( pass1 ) , & sampass ) & &
change_oem_password ( sampass , pass2 , False ) )
2000-04-19 11:00:22 +04:00
{
SSVAL ( * rparam , 0 , NERR_Success ) ;
/*
* If unix password sync was requested , attempt to change
* the / etc / passwd database also . Return failure if this cannot
* be done .
*/
if ( lp_unix_password_sync ( ) & & ! chgpasswd ( user , pass1 , saved_pass2 , False ) )
SSVAL ( * rparam , 0 , NERR_badpass ) ;
}
2001-05-07 18:04:46 +04:00
pdb_free_sam ( sampass ) ;
2001-05-04 19:44:27 +04:00
}
2000-04-19 11:00:22 +04:00
/*
* If the above failed , attempt the plaintext password change .
* This tests against the / etc / passwd database only .
*/
if ( SVAL ( * rparam , 0 ) ! = NERR_Success )
{
2001-07-08 18:10:30 +04:00
if ( password_ok ( user , pass1 , strlen ( pass1 ) ) & &
2000-04-19 11:00:22 +04:00
chgpasswd ( user , pass1 , pass2 , False ) )
{
SSVAL ( * rparam , 0 , NERR_Success ) ;
}
}
/*
* If the plaintext change failed , attempt
* the old encrypted method . NT will generate this
* after trying the samr method . Note that this
* method is done as a last resort as this
* password change method loses the NT password hash
* and cannot change the UNIX password as no plaintext
* is received .
*/
if ( SVAL ( * rparam , 0 ) ! = NERR_Success )
{
2000-11-14 02:03:34 +03:00
SAM_ACCOUNT * hnd = NULL ;
2000-04-19 11:00:22 +04:00
2000-11-14 02:03:34 +03:00
if ( check_lanman_password ( user , ( unsigned char * ) pass1 , ( unsigned char * ) pass2 , & hnd ) & &
change_lanman_password ( hnd , ( unsigned char * ) pass1 , ( unsigned char * ) pass2 ) )
2000-04-19 11:00:22 +04:00
{
SSVAL ( * rparam , 0 , NERR_Success ) ;
}
2001-05-07 18:04:46 +04:00
pdb_free_sam ( hnd ) ;
2000-04-19 11:00:22 +04:00
}
2001-05-04 19:44:27 +04:00
2000-04-19 11:00:22 +04:00
memset ( ( char * ) pass1 , ' \0 ' , sizeof ( fstring ) ) ;
memset ( ( char * ) pass2 , ' \0 ' , sizeof ( fstring ) ) ;
return ( True ) ;
}
/****************************************************************************
Set the user password ( SamOEM version - gets plaintext ) .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static BOOL api_SamOEMChangePassword ( connection_struct * conn , uint16 vuid , char * param , char * data ,
int mdrcnt , int mprcnt ,
char * * rdata , char * * rparam ,
int * rdata_len , int * rparam_len )
{
fstring user ;
char * p = param + 2 ;
* rparam_len = 2 ;
* rparam = REALLOC ( * rparam , * rparam_len ) ;
* rdata_len = 0 ;
SSVAL ( * rparam , 0 , NERR_badpass ) ;
/*
* Check the parameter definition is correct .
*/
if ( ! strequal ( param + 2 , " zsT " ) ) {
DEBUG ( 0 , ( " api_SamOEMChangePassword: Invalid parameter string %s \n " , param + 2 ) ) ;
return False ;
}
p = skip_string ( p , 1 ) ;
if ( ! strequal ( p , " B516B16 " ) ) {
DEBUG ( 0 , ( " api_SamOEMChangePassword: Invalid data parameter string %s \n " , p ) ) ;
return False ;
}
p = skip_string ( p , 1 ) ;
2001-07-04 11:15:53 +04:00
p + = pull_ascii_fstring ( user , p ) ;
2000-04-19 11:00:22 +04:00
DEBUG ( 3 , ( " api_SamOEMChangePassword: Change password for <%s> \n " , user ) ) ;
/*
* Pass the user through the NT - > unix user mapping
* function .
*/
( void ) map_username ( user ) ;
/*
* Do any UNIX username case mangling .
*/
( void ) Get_Pwnam ( user , True ) ;
if ( pass_oem_change ( user , ( uchar * ) data , ( uchar * ) & data [ 516 ] , NULL , NULL ) )
{
SSVAL ( * rparam , 0 , NERR_Success ) ;
}
return ( True ) ;
}
/****************************************************************************
delete a print job
Form : < W > < >
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static BOOL api_RDosPrintJobDel ( connection_struct * conn , uint16 vuid , char * param , char * data ,
int mdrcnt , int mprcnt ,
char * * rdata , char * * rparam ,
int * rdata_len , int * rparam_len )
{
int function = SVAL ( param , 0 ) ;
char * str1 = param + 2 ;
char * str2 = skip_string ( str1 , 1 ) ;
char * p = skip_string ( str2 , 1 ) ;
int jobid , errcode ;
2000-07-10 09:08:21 +04:00
extern struct current_user current_user ;
2001-09-04 11:13:01 +04:00
WERROR werr = WERR_OK ;
2000-04-19 11:00:22 +04:00
jobid = SVAL ( p , 0 ) ;
/* check it's a supported varient */
if ( ! ( strcsequal ( str1 , " W " ) & & strcsequal ( str2 , " " ) ) )
return ( False ) ;
* rparam_len = 4 ;
* rparam = REALLOC ( * rparam , * rparam_len ) ;
* rdata_len = 0 ;
if ( ! print_job_exists ( jobid ) ) {
errcode = NERR_JobNotFound ;
goto out ;
}
errcode = NERR_notsupported ;
switch ( function ) {
case 81 : /* delete */
2001-09-04 11:13:01 +04:00
if ( print_job_delete ( & current_user , jobid , & werr ) )
2000-07-10 09:08:21 +04:00
errcode = NERR_Success ;
2000-04-19 11:00:22 +04:00
break ;
case 82 : /* pause */
2001-09-04 11:13:01 +04:00
if ( print_job_pause ( & current_user , jobid , & werr ) )
2000-07-10 09:08:21 +04:00
errcode = NERR_Success ;
2000-04-19 11:00:22 +04:00
break ;
case 83 : /* resume */
2001-09-04 11:13:01 +04:00
if ( print_job_resume ( & current_user , jobid , & werr ) )
2000-07-10 09:08:21 +04:00
errcode = NERR_Success ;
2000-04-19 11:00:22 +04:00
break ;
}
2001-09-04 11:13:01 +04:00
if ( ! W_ERROR_IS_OK ( werr ) )
errcode = W_ERROR_V ( werr ) ;
2000-04-19 11:00:22 +04:00
out :
SSVAL ( * rparam , 0 , errcode ) ;
SSVAL ( * rparam , 2 , 0 ) ; /* converter word */
return ( True ) ;
}
/****************************************************************************
Purge a print queue - or pause or resume it .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2000-10-18 10:22:20 +04:00
static BOOL api_WPrintQueueCtrl ( connection_struct * conn , uint16 vuid , char * param , char * data ,
2000-04-19 11:00:22 +04:00
int mdrcnt , int mprcnt ,
char * * rdata , char * * rparam ,
int * rdata_len , int * rparam_len )
{
int function = SVAL ( param , 0 ) ;
char * str1 = param + 2 ;
char * str2 = skip_string ( str1 , 1 ) ;
char * QueueName = skip_string ( str2 , 1 ) ;
int errcode = NERR_notsupported ;
int snum ;
2001-09-04 11:13:01 +04:00
WERROR werr = WERR_OK ;
2001-01-22 19:59:24 +03:00
extern struct current_user current_user ;
2000-04-19 11:00:22 +04:00
/* check it's a supported varient */
if ( ! ( strcsequal ( str1 , " z " ) & & strcsequal ( str2 , " " ) ) )
return ( False ) ;
* rparam_len = 4 ;
* rparam = REALLOC ( * rparam , * rparam_len ) ;
* rdata_len = 0 ;
snum = print_queue_snum ( QueueName ) ;
if ( snum = = - 1 ) {
errcode = NERR_JobNotFound ;
goto out ;
}
switch ( function ) {
case 74 : /* Pause queue */
2001-09-04 11:13:01 +04:00
if ( print_queue_pause ( & current_user , snum , & werr ) ) errcode = NERR_Success ;
2000-04-19 11:00:22 +04:00
break ;
case 75 : /* Resume queue */
2001-09-04 11:13:01 +04:00
if ( print_queue_resume ( & current_user , snum , & werr ) ) errcode = NERR_Success ;
2000-04-19 11:00:22 +04:00
break ;
case 103 : /* Purge */
2001-09-04 11:13:01 +04:00
if ( print_queue_purge ( & current_user , snum , & werr ) ) errcode = NERR_Success ;
2000-04-19 11:00:22 +04:00
break ;
}
2001-09-04 11:13:01 +04:00
if ( ! W_ERROR_IS_OK ( werr ) ) errcode = W_ERROR_V ( werr ) ;
2000-04-19 11:00:22 +04:00
out :
SSVAL ( * rparam , 0 , errcode ) ;
SSVAL ( * rparam , 2 , 0 ) ; /* converter word */
return ( True ) ;
}
/****************************************************************************
set the property of a print job ( undocumented ? )
? function = 0xb - > set name of print job
? function = 0x6 - > move print job up / down
Form : < WWsTP > < WWzWWDDzzzzzzzzzzlz >
or < WWsTP > < WB21BB16B10zWWzDDz >
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static int check_printjob_info ( struct pack_desc * desc ,
int uLevel , char * id )
{
desc - > subformat = NULL ;
switch ( uLevel ) {
case 0 : desc - > format = " W " ; break ;
case 1 : desc - > format = " WB21BB16B10zWWzDDz " ; break ;
case 2 : desc - > format = " WWzWWDDzz " ; break ;
case 3 : desc - > format = " WWzWWDDzzzzzzzzzzlz " ; break ;
2001-03-23 06:12:58 +03:00
case 4 : desc - > format = " WWzWWDDzzzzzDDDDDDD " ; break ;
2000-04-19 11:00:22 +04:00
default : return False ;
}
if ( strcmp ( desc - > format , id ) ! = 0 ) return False ;
return True ;
}
static BOOL api_PrintJobInfo ( connection_struct * conn , uint16 vuid , char * param , char * data ,
int mdrcnt , int mprcnt ,
char * * rdata , char * * rparam ,
int * rdata_len , int * rparam_len )
{
struct pack_desc desc ;
char * str1 = param + 2 ;
char * str2 = skip_string ( str1 , 1 ) ;
char * p = skip_string ( str2 , 1 ) ;
int jobid ;
int uLevel = SVAL ( p , 2 ) ;
int function = SVAL ( p , 4 ) ;
int place , errcode ;
jobid = SVAL ( p , 0 ) ;
* rparam_len = 4 ;
* rparam = REALLOC ( * rparam , * rparam_len ) ;
* rdata_len = 0 ;
/* check it's a supported varient */
if ( ( strcmp ( str1 , " WWsTP " ) ) | |
( ! check_printjob_info ( & desc , uLevel , str2 ) ) )
return ( False ) ;
if ( ! print_job_exists ( jobid ) ) {
errcode = NERR_JobNotFound ;
goto out ;
}
errcode = NERR_notsupported ;
switch ( function ) {
case 0x6 :
/* change job place in the queue,
data gives the new place */
place = SVAL ( data , 0 ) ;
if ( print_job_set_place ( jobid , place ) ) {
errcode = NERR_Success ;
}
break ;
case 0xb :
/* change print job name, data gives the name */
if ( print_job_set_name ( jobid , data ) ) {
errcode = NERR_Success ;
}
break ;
default :
return False ;
}
out :
SSVALS ( * rparam , 0 , errcode ) ;
SSVAL ( * rparam , 2 , 0 ) ; /* converter word */
return ( True ) ;
}
/****************************************************************************
get info about the server
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static BOOL api_RNetServerGetInfo ( connection_struct * conn , uint16 vuid , char * param , char * data ,
int mdrcnt , int mprcnt ,
char * * rdata , char * * rparam ,
int * rdata_len , int * rparam_len )
{
char * str1 = param + 2 ;
char * str2 = skip_string ( str1 , 1 ) ;
char * p = skip_string ( str2 , 1 ) ;
int uLevel = SVAL ( p , 0 ) ;
char * p2 ;
int struct_len ;
DEBUG ( 4 , ( " NetServerGetInfo level %d \n " , uLevel ) ) ;
/* check it's a supported varient */
if ( ! prefix_ok ( str1 , " WrLh " ) ) return False ;
switch ( uLevel ) {
case 0 :
if ( strcmp ( str2 , " B16 " ) ! = 0 ) return False ;
struct_len = 16 ;
break ;
case 1 :
if ( strcmp ( str2 , " B16BBDz " ) ! = 0 ) return False ;
struct_len = 26 ;
break ;
case 2 :
if ( strcmp ( str2 , " B16BBDzDDDWWzWWWWWWWBB21zWWWWWWWWWWWWWWWWWWWWWWz " )
! = 0 ) return False ;
struct_len = 134 ;
break ;
case 3 :
if ( strcmp ( str2 , " B16BBDzDDDWWzWWWWWWWBB21zWWWWWWWWWWWWWWWWWWWWWWzDWz " )
! = 0 ) return False ;
struct_len = 144 ;
break ;
case 20 :
if ( strcmp ( str2 , " DN " ) ! = 0 ) return False ;
struct_len = 6 ;
break ;
case 50 :
if ( strcmp ( str2 , " B16BBDzWWzzz " ) ! = 0 ) return False ;
struct_len = 42 ;
break ;
default : return False ;
}
* rdata_len = mdrcnt ;
* rdata = REALLOC ( * rdata , * rdata_len ) ;
p = * rdata ;
p2 = p + struct_len ;
if ( uLevel ! = 20 ) {
2001-03-31 17:48:24 +04:00
srvstr_push ( NULL , p , local_machine , 16 ,
2001-07-04 11:15:53 +04:00
STR_ASCII | STR_UPPER | STR_TERMINATE ) ;
2000-04-19 11:00:22 +04:00
}
p + = 16 ;
if ( uLevel > 0 )
{
struct srv_info_struct * servers = NULL ;
int i , count ;
pstring comment ;
uint32 servertype = lp_default_server_announce ( ) ;
pstrcpy ( comment , string_truncate ( lp_serverstring ( ) , MAX_SERVER_STRING_LENGTH ) ) ;
if ( ( count = get_server_info ( SV_TYPE_ALL , & servers , global_myworkgroup ) ) > 0 ) {
for ( i = 0 ; i < count ; i + + )
if ( strequal ( servers [ i ] . name , local_machine ) )
{
servertype = servers [ i ] . type ;
pstrcpy ( comment , servers [ i ] . comment ) ;
}
}
2001-09-17 15:25:41 +04:00
SAFE_FREE ( servers ) ;
2000-04-19 11:00:22 +04:00
SCVAL ( p , 0 , lp_major_announce_version ( ) ) ;
SCVAL ( p , 1 , lp_minor_announce_version ( ) ) ;
SIVAL ( p , 2 , servertype ) ;
if ( mdrcnt = = struct_len ) {
SIVAL ( p , 6 , 0 ) ;
} else {
SIVAL ( p , 6 , PTR_DIFF ( p2 , * rdata ) ) ;
standard_sub_conn ( conn , comment ) ;
StrnCpy ( p2 , comment , MAX ( mdrcnt - struct_len , 0 ) ) ;
p2 = skip_string ( p2 , 1 ) ;
}
}
if ( uLevel > 1 )
{
return False ; /* not yet implemented */
}
* rdata_len = PTR_DIFF ( p2 , * rdata ) ;
* rparam_len = 6 ;
* rparam = REALLOC ( * rparam , * rparam_len ) ;
SSVAL ( * rparam , 0 , NERR_Success ) ;
SSVAL ( * rparam , 2 , 0 ) ; /* converter word */
SSVAL ( * rparam , 4 , * rdata_len ) ;
return ( True ) ;
}
/****************************************************************************
get info about the server
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static BOOL api_NetWkstaGetInfo ( connection_struct * conn , uint16 vuid , char * param , char * data ,
int mdrcnt , int mprcnt ,
char * * rdata , char * * rparam ,
int * rdata_len , int * rparam_len )
{
char * str1 = param + 2 ;
char * str2 = skip_string ( str1 , 1 ) ;
char * p = skip_string ( str2 , 1 ) ;
char * p2 ;
2001-01-23 04:52:30 +03:00
extern userdom_struct current_user_info ;
2000-04-19 11:00:22 +04:00
int level = SVAL ( p , 0 ) ;
DEBUG ( 4 , ( " NetWkstaGetInfo level %d \n " , level ) ) ;
* rparam_len = 6 ;
* rparam = REALLOC ( * rparam , * rparam_len ) ;
/* check it's a supported varient */
if ( ! ( level = = 10 & & strcsequal ( str1 , " WrLh " ) & & strcsequal ( str2 , " zzzBBzz " ) ) )
return ( False ) ;
* rdata_len = mdrcnt + 1024 ;
* rdata = REALLOC ( * rdata , * rdata_len ) ;
SSVAL ( * rparam , 0 , NERR_Success ) ;
SSVAL ( * rparam , 2 , 0 ) ; /* converter word */
p = * rdata ;
p2 = p + 22 ;
SIVAL ( p , 0 , PTR_DIFF ( p2 , * rdata ) ) ; /* host name */
pstrcpy ( p2 , local_machine ) ;
strupper ( p2 ) ;
p2 = skip_string ( p2 , 1 ) ;
p + = 4 ;
SIVAL ( p , 0 , PTR_DIFF ( p2 , * rdata ) ) ;
2001-01-23 04:52:30 +03:00
pstrcpy ( p2 , current_user_info . smb_name ) ;
2000-04-19 11:00:22 +04:00
p2 = skip_string ( p2 , 1 ) ;
p + = 4 ;
SIVAL ( p , 0 , PTR_DIFF ( p2 , * rdata ) ) ; /* login domain */
pstrcpy ( p2 , global_myworkgroup ) ;
strupper ( p2 ) ;
p2 = skip_string ( p2 , 1 ) ;
p + = 4 ;
SCVAL ( p , 0 , lp_major_announce_version ( ) ) ; /* system version - e.g 4 in 4.1 */
SCVAL ( p , 1 , lp_minor_announce_version ( ) ) ; /* system version - e.g .1 in 4.1 */
p + = 2 ;
SIVAL ( p , 0 , PTR_DIFF ( p2 , * rdata ) ) ;
pstrcpy ( p2 , global_myworkgroup ) ; /* don't know. login domain?? */
p2 = skip_string ( p2 , 1 ) ;
p + = 4 ;
SIVAL ( p , 0 , PTR_DIFF ( p2 , * rdata ) ) ; /* don't know */
pstrcpy ( p2 , " " ) ;
p2 = skip_string ( p2 , 1 ) ;
p + = 4 ;
* rdata_len = PTR_DIFF ( p2 , * rdata ) ;
SSVAL ( * rparam , 4 , * rdata_len ) ;
return ( True ) ;
}
/****************************************************************************
get info about a user
struct user_info_11 {
char usri11_name [ 21 ] ; 0 - 20
char usri11_pad ; 21
char * usri11_comment ; 22 - 25
char * usri11_usr_comment ; 26 - 29
unsigned short usri11_priv ; 30 - 31
unsigned long usri11_auth_flags ; 32 - 35
long usri11_password_age ; 36 - 39
char * usri11_homedir ; 40 - 43
char * usri11_parms ; 44 - 47
long usri11_last_logon ; 48 - 51
long usri11_last_logoff ; 52 - 55
unsigned short usri11_bad_pw_count ; 56 - 57
unsigned short usri11_num_logons ; 58 - 59
char * usri11_logon_server ; 60 - 63
unsigned short usri11_country_code ; 64 - 65
char * usri11_workstations ; 66 - 69
unsigned long usri11_max_storage ; 70 - 73
unsigned short usri11_units_per_week ; 74 - 75
unsigned char * usri11_logon_hours ; 76 - 79
unsigned short usri11_code_page ; 80 - 81
} ;
where :
usri11_name specifies the user name for which information is retireved
usri11_pad aligns the next data structure element to a word boundary
usri11_comment is a null terminated ASCII comment
usri11_user_comment is a null terminated ASCII comment about the user
usri11_priv specifies the level of the privilege assigned to the user .
The possible values are :
Name Value Description
USER_PRIV_GUEST 0 Guest privilege
USER_PRIV_USER 1 User privilege
USER_PRV_ADMIN 2 Administrator privilege
usri11_auth_flags specifies the account operator privileges . The
possible values are :
Name Value Description
AF_OP_PRINT 0 Print operator
2000-08-18 00:05:05 +04:00
Leach , Naik [ Page 28 ]
2000-04-19 11:00:22 +04:00
INTERNET - DRAFT CIFS Remote Admin Protocol January 10 , 1997
AF_OP_COMM 1 Communications operator
AF_OP_SERVER 2 Server operator
AF_OP_ACCOUNTS 3 Accounts operator
usri11_password_age specifies how many seconds have elapsed since the
password was last changed .
usri11_home_dir points to a null terminated ASCII string that contains
the path name of the user ' s home directory .
usri11_parms points to a null terminated ASCII string that is set
aside for use by applications .
usri11_last_logon specifies the time when the user last logged on .
This value is stored as the number of seconds elapsed since
00 : 00 : 00 , January 1 , 1970.
usri11_last_logoff specifies the time when the user last logged off .
This value is stored as the number of seconds elapsed since
00 : 00 : 00 , January 1 , 1970. A value of 0 means the last logoff
time is unknown .
usri11_bad_pw_count specifies the number of incorrect passwords
entered since the last successful logon .
usri11_log1_num_logons specifies the number of times this user has
logged on . A value of - 1 means the number of logons is unknown .
usri11_logon_server points to a null terminated ASCII string that
contains the name of the server to which logon requests are sent .
A null string indicates logon requests should be sent to the
domain controller .
usri11_country_code specifies the country code for the user ' s language
of choice .
usri11_workstations points to a null terminated ASCII string that
contains the names of workstations the user may log on from .
There may be up to 8 workstations , with the names separated by
commas . A null strings indicates there are no restrictions .
usri11_max_storage specifies the maximum amount of disk space the user
can occupy . A value of 0xffffffff indicates there are no
restrictions .
usri11_units_per_week specifies the equal number of time units into
which a week is divided . This value must be equal to 168.
usri11_logon_hours points to a 21 byte ( 168 bits ) string that
specifies the time during which the user can log on . Each bit
represents one unique hour in a week . The first bit ( bit 0 , word
0 ) is Sunday , 0 : 00 to 0 : 59 , the second bit ( bit 1 , word 0 ) is
2000-08-18 00:05:05 +04:00
Leach , Naik [ Page 29 ]
2000-04-19 11:00:22 +04:00
INTERNET - DRAFT CIFS Remote Admin Protocol January 10 , 1997
Sunday , 1 : 00 to 1 : 59 and so on . A null pointer indicates there
are no restrictions .
usri11_code_page specifies the code page for the user ' s language of
choice
All of the pointers in this data structure need to be treated
specially . The pointer is a 32 bit pointer . The higher 16 bits need
to be ignored . The converter word returned in the parameters section
needs to be subtracted from the lower 16 bits to calculate an offset
into the return buffer where this ASCII string resides .
There is no auxiliary data in the response .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
# define usri11_name 0
# define usri11_pad 21
# define usri11_comment 22
# define usri11_usr_comment 26
# define usri11_full_name 30
# define usri11_priv 34
# define usri11_auth_flags 36
# define usri11_password_age 40
# define usri11_homedir 44
# define usri11_parms 48
# define usri11_last_logon 52
# define usri11_last_logoff 56
# define usri11_bad_pw_count 60
# define usri11_num_logons 62
# define usri11_logon_server 64
# define usri11_country_code 68
# define usri11_workstations 70
# define usri11_max_storage 74
# define usri11_units_per_week 78
# define usri11_logon_hours 80
# define usri11_code_page 84
# define usri11_end 86
# define USER_PRIV_GUEST 0
# define USER_PRIV_USER 1
# define USER_PRIV_ADMIN 2
# define AF_OP_PRINT 0
# define AF_OP_COMM 1
# define AF_OP_SERVER 2
# define AF_OP_ACCOUNTS 3
static BOOL api_RNetUserGetInfo ( connection_struct * conn , uint16 vuid , char * param , char * data ,
int mdrcnt , int mprcnt ,
char * * rdata , char * * rparam ,
int * rdata_len , int * rparam_len )
{
char * str1 = param + 2 ;
char * str2 = skip_string ( str1 , 1 ) ;
char * UserName = skip_string ( str2 , 1 ) ;
char * p = skip_string ( UserName , 1 ) ;
int uLevel = SVAL ( p , 0 ) ;
char * p2 ;
/* get NIS home of a previously validated user - simeon */
/* With share level security vuid will always be zero.
Don ' t depend on vuser being non - null ! ! . JRA */
user_struct * vuser = get_valid_user_struct ( vuid ) ;
if ( vuser ! = NULL )
2000-05-02 17:55:42 +04:00
DEBUG ( 3 , ( " Username of UID %d is %s \n " , ( int ) vuser - > uid ,
vuser - > user . unix_name ) ) ;
2000-04-19 11:00:22 +04:00
* rparam_len = 6 ;
* rparam = REALLOC ( * rparam , * rparam_len ) ;
DEBUG ( 4 , ( " RNetUserGetInfo level=%d \n " , uLevel ) ) ;
/* check it's a supported variant */
if ( strcmp ( str1 , " zWrLh " ) ! = 0 ) return False ;
switch ( uLevel )
{
case 0 : p2 = " B21 " ; break ;
case 1 : p2 = " B21BB16DWzzWz " ; break ;
case 2 : p2 = " B21BB16DWzzWzDzzzzDDDDWb21WWzWW " ; break ;
case 10 : p2 = " B21Bzzz " ; break ;
case 11 : p2 = " B21BzzzWDDzzDDWWzWzDWb21W " ; break ;
default : return False ;
}
if ( strcmp ( p2 , str2 ) ! = 0 ) return False ;
* rdata_len = mdrcnt + 1024 ;
* rdata = REALLOC ( * rdata , * rdata_len ) ;
SSVAL ( * rparam , 0 , NERR_Success ) ;
SSVAL ( * rparam , 2 , 0 ) ; /* converter word */
p = * rdata ;
p2 = p + usri11_end ;
memset ( p , 0 , 21 ) ;
fstrcpy ( p + usri11_name , UserName ) ; /* 21 bytes - user name */
if ( uLevel > 0 )
{
SCVAL ( p , usri11_pad , 0 ) ; /* padding - 1 byte */
* p2 = 0 ;
}
if ( uLevel > = 10 )
{
SIVAL ( p , usri11_comment , PTR_DIFF ( p2 , p ) ) ; /* comment */
pstrcpy ( p2 , " Comment " ) ;
p2 = skip_string ( p2 , 1 ) ;
SIVAL ( p , usri11_usr_comment , PTR_DIFF ( p2 , p ) ) ; /* user_comment */
pstrcpy ( p2 , " UserComment " ) ;
p2 = skip_string ( p2 , 1 ) ;
/* EEK! the cifsrap.txt doesn't have this in!!!! */
SIVAL ( p , usri11_full_name , PTR_DIFF ( p2 , p ) ) ; /* full name */
2000-05-04 11:59:34 +04:00
pstrcpy ( p2 , ( ( vuser ! = NULL ) ? vuser - > user . full_name : UserName ) ) ;
2000-04-19 11:00:22 +04:00
p2 = skip_string ( p2 , 1 ) ;
}
if ( uLevel = = 11 ) /* modelled after NTAS 3.51 reply */
{
SSVAL ( p , usri11_priv , conn - > admin_user ? USER_PRIV_ADMIN : USER_PRIV_USER ) ;
SIVAL ( p , usri11_auth_flags , AF_OP_PRINT ) ; /* auth flags */
SIVALS ( p , usri11_password_age , - 1 ) ; /* password age */
SIVAL ( p , usri11_homedir , PTR_DIFF ( p2 , p ) ) ; /* home dir */
pstrcpy ( p2 , lp_logon_home ( ) ) ;
2000-10-30 22:35:43 +03:00
standard_sub_conn ( conn , p2 ) ;
2000-04-19 11:00:22 +04:00
p2 = skip_string ( p2 , 1 ) ;
SIVAL ( p , usri11_parms , PTR_DIFF ( p2 , p ) ) ; /* parms */
pstrcpy ( p2 , " " ) ;
p2 = skip_string ( p2 , 1 ) ;
SIVAL ( p , usri11_last_logon , 0 ) ; /* last logon */
SIVAL ( p , usri11_last_logoff , 0 ) ; /* last logoff */
SSVALS ( p , usri11_bad_pw_count , - 1 ) ; /* bad pw counts */
SSVALS ( p , usri11_num_logons , - 1 ) ; /* num logons */
SIVAL ( p , usri11_logon_server , PTR_DIFF ( p2 , p ) ) ; /* logon server */
pstrcpy ( p2 , " \\ \\ * " ) ;
p2 = skip_string ( p2 , 1 ) ;
SSVAL ( p , usri11_country_code , 0 ) ; /* country code */
SIVAL ( p , usri11_workstations , PTR_DIFF ( p2 , p ) ) ; /* workstations */
pstrcpy ( p2 , " " ) ;
p2 = skip_string ( p2 , 1 ) ;
SIVALS ( p , usri11_max_storage , - 1 ) ; /* max storage */
SSVAL ( p , usri11_units_per_week , 168 ) ; /* units per week */
SIVAL ( p , usri11_logon_hours , PTR_DIFF ( p2 , p ) ) ; /* logon hours */
/* a simple way to get logon hours at all times. */
memset ( p2 , 0xff , 21 ) ;
SCVAL ( p2 , 21 , 0 ) ; /* fix zero termination */
p2 = skip_string ( p2 , 1 ) ;
SSVAL ( p , usri11_code_page , 0 ) ; /* code page */
}
if ( uLevel = = 1 | | uLevel = = 2 )
{
memset ( p + 22 , ' ' , 16 ) ; /* password */
SIVALS ( p , 38 , - 1 ) ; /* password age */
SSVAL ( p , 42 ,
conn - > admin_user ? USER_PRIV_ADMIN : USER_PRIV_USER ) ;
SIVAL ( p , 44 , PTR_DIFF ( p2 , * rdata ) ) ; /* home dir */
pstrcpy ( p2 , lp_logon_home ( ) ) ;
2000-10-30 22:35:43 +03:00
standard_sub_conn ( conn , p2 ) ;
2000-04-19 11:00:22 +04:00
p2 = skip_string ( p2 , 1 ) ;
SIVAL ( p , 48 , PTR_DIFF ( p2 , * rdata ) ) ; /* comment */
* p2 + + = 0 ;
SSVAL ( p , 52 , 0 ) ; /* flags */
SIVAL ( p , 54 , PTR_DIFF ( p2 , * rdata ) ) ; /* script_path */
pstrcpy ( p2 , lp_logon_script ( ) ) ;
standard_sub_conn ( conn , p2 ) ;
p2 = skip_string ( p2 , 1 ) ;
if ( uLevel = = 2 )
{
SIVAL ( p , 60 , 0 ) ; /* auth_flags */
SIVAL ( p , 64 , PTR_DIFF ( p2 , * rdata ) ) ; /* full_name */
2000-05-04 11:59:34 +04:00
pstrcpy ( p2 , ( ( vuser ! = NULL ) ? vuser - > user . full_name : UserName ) ) ;
2000-04-19 11:00:22 +04:00
p2 = skip_string ( p2 , 1 ) ;
SIVAL ( p , 68 , 0 ) ; /* urs_comment */
SIVAL ( p , 72 , PTR_DIFF ( p2 , * rdata ) ) ; /* parms */
pstrcpy ( p2 , " " ) ;
p2 = skip_string ( p2 , 1 ) ;
SIVAL ( p , 76 , 0 ) ; /* workstations */
SIVAL ( p , 80 , 0 ) ; /* last_logon */
SIVAL ( p , 84 , 0 ) ; /* last_logoff */
SIVALS ( p , 88 , - 1 ) ; /* acct_expires */
SIVALS ( p , 92 , - 1 ) ; /* max_storage */
SSVAL ( p , 96 , 168 ) ; /* units_per_week */
SIVAL ( p , 98 , PTR_DIFF ( p2 , * rdata ) ) ; /* logon_hours */
memset ( p2 , - 1 , 21 ) ;
p2 + = 21 ;
SSVALS ( p , 102 , - 1 ) ; /* bad_pw_count */
SSVALS ( p , 104 , - 1 ) ; /* num_logons */
SIVAL ( p , 106 , PTR_DIFF ( p2 , * rdata ) ) ; /* logon_server */
pstrcpy ( p2 , " \\ \\ %L " ) ;
standard_sub_conn ( conn , p2 ) ;
p2 = skip_string ( p2 , 1 ) ;
SSVAL ( p , 110 , 49 ) ; /* country_code */
SSVAL ( p , 112 , 860 ) ; /* code page */
}
}
* rdata_len = PTR_DIFF ( p2 , * rdata ) ;
SSVAL ( * rparam , 4 , * rdata_len ) ; /* is this right?? */
return ( True ) ;
}
/*******************************************************************
get groups that a user is a member of
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static BOOL api_NetUserGetGroups ( connection_struct * conn , uint16 vuid , char * param , char * data ,
int mdrcnt , int mprcnt ,
char * * rdata , char * * rparam ,
int * rdata_len , int * rparam_len )
{
char * str1 = param + 2 ;
char * str2 = skip_string ( str1 , 1 ) ;
char * UserName = skip_string ( str2 , 1 ) ;
char * p = skip_string ( UserName , 1 ) ;
int uLevel = SVAL ( p , 0 ) ;
char * p2 ;
int count = 0 ;
* rparam_len = 8 ;
* rparam = REALLOC ( * rparam , * rparam_len ) ;
/* check it's a supported varient */
if ( strcmp ( str1 , " zWrLeh " ) ! = 0 ) return False ;
switch ( uLevel ) {
case 0 : p2 = " B21 " ; break ;
default : return False ;
}
if ( strcmp ( p2 , str2 ) ! = 0 ) return False ;
* rdata_len = mdrcnt + 1024 ;
* rdata = REALLOC ( * rdata , * rdata_len ) ;
SSVAL ( * rparam , 0 , NERR_Success ) ;
SSVAL ( * rparam , 2 , 0 ) ; /* converter word */
p = * rdata ;
/* XXXX we need a real SAM database some day */
pstrcpy ( p , " Users " ) ; p + = 21 ; count + + ;
pstrcpy ( p , " Domain Users " ) ; p + = 21 ; count + + ;
pstrcpy ( p , " Guests " ) ; p + = 21 ; count + + ;
pstrcpy ( p , " Domain Guests " ) ; p + = 21 ; count + + ;
* rdata_len = PTR_DIFF ( p , * rdata ) ;
SSVAL ( * rparam , 4 , count ) ; /* is this right?? */
SSVAL ( * rparam , 6 , count ) ; /* is this right?? */
return ( True ) ;
}
static BOOL api_WWkstaUserLogon ( connection_struct * conn , uint16 vuid , char * param , char * data ,
int mdrcnt , int mprcnt ,
char * * rdata , char * * rparam ,
int * rdata_len , int * rparam_len )
{
char * str1 = param + 2 ;
char * str2 = skip_string ( str1 , 1 ) ;
char * p = skip_string ( str2 , 1 ) ;
int uLevel ;
struct pack_desc desc ;
char * name ;
uLevel = SVAL ( p , 0 ) ;
name = p + 2 ;
memset ( ( char * ) & desc , ' \0 ' , sizeof ( desc ) ) ;
DEBUG ( 3 , ( " WWkstaUserLogon uLevel=%d name=%s \n " , uLevel , name ) ) ;
/* check it's a supported varient */
if ( strcmp ( str1 , " OOWb54WrLh " ) ! = 0 ) return False ;
if ( uLevel ! = 1 | | strcmp ( str2 , " WB21BWDWWDDDDDDDzzzD " ) ! = 0 ) return False ;
if ( mdrcnt > 0 ) * rdata = REALLOC ( * rdata , mdrcnt ) ;
desc . base = * rdata ;
desc . buflen = mdrcnt ;
desc . subformat = NULL ;
desc . format = str2 ;
if ( init_package ( & desc , 1 , 0 ) )
{
PACKI ( & desc , " W " , 0 ) ; /* code */
PACKS ( & desc , " B21 " , name ) ; /* eff. name */
PACKS ( & desc , " B " , " " ) ; /* pad */
PACKI ( & desc , " W " ,
conn - > admin_user ? USER_PRIV_ADMIN : USER_PRIV_USER ) ;
PACKI ( & desc , " D " , 0 ) ; /* auth flags XXX */
PACKI ( & desc , " W " , 0 ) ; /* num logons */
PACKI ( & desc , " W " , 0 ) ; /* bad pw count */
PACKI ( & desc , " D " , 0 ) ; /* last logon */
PACKI ( & desc , " D " , - 1 ) ; /* last logoff */
PACKI ( & desc , " D " , - 1 ) ; /* logoff time */
PACKI ( & desc , " D " , - 1 ) ; /* kickoff time */
PACKI ( & desc , " D " , 0 ) ; /* password age */
PACKI ( & desc , " D " , 0 ) ; /* password can change */
PACKI ( & desc , " D " , - 1 ) ; /* password must change */
{
fstring mypath ;
fstrcpy ( mypath , " \\ \\ " ) ;
fstrcat ( mypath , local_machine ) ;
strupper ( mypath ) ;
PACKS ( & desc , " z " , mypath ) ; /* computer */
}
PACKS ( & desc , " z " , global_myworkgroup ) ; /* domain */
/* JHT - By calling lp_logon_script() and standard_sub() we have */
/* made sure all macros are fully substituted and available */
{
pstring logon_script ;
pstrcpy ( logon_script , lp_logon_script ( ) ) ;
standard_sub_conn ( conn , logon_script ) ;
PACKS ( & desc , " z " , logon_script ) ; /* script path */
}
/* End of JHT mods */
PACKI ( & desc , " D " , 0x00000000 ) ; /* reserved */
}
* rdata_len = desc . usedlen ;
* rparam_len = 6 ;
* rparam = REALLOC ( * rparam , * rparam_len ) ;
SSVALS ( * rparam , 0 , desc . errcode ) ;
SSVAL ( * rparam , 2 , 0 ) ;
SSVAL ( * rparam , 4 , desc . neededlen ) ;
DEBUG ( 4 , ( " WWkstaUserLogon: errorcode %d \n " , desc . errcode ) ) ;
return ( True ) ;
}
/****************************************************************************
api_WAccessGetUserPerms
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static BOOL api_WAccessGetUserPerms ( connection_struct * conn , uint16 vuid , char * param , char * data ,
int mdrcnt , int mprcnt ,
char * * rdata , char * * rparam ,
int * rdata_len , int * rparam_len )
{
char * str1 = param + 2 ;
char * str2 = skip_string ( str1 , 1 ) ;
char * user = skip_string ( str2 , 1 ) ;
char * resource = skip_string ( user , 1 ) ;
DEBUG ( 3 , ( " WAccessGetUserPerms user=%s resource=%s \n " , user , resource ) ) ;
/* check it's a supported varient */
if ( strcmp ( str1 , " zzh " ) ! = 0 ) return False ;
if ( strcmp ( str2 , " " ) ! = 0 ) return False ;
* rparam_len = 6 ;
* rparam = REALLOC ( * rparam , * rparam_len ) ;
SSVALS ( * rparam , 0 , 0 ) ; /* errorcode */
SSVAL ( * rparam , 2 , 0 ) ; /* converter word */
SSVAL ( * rparam , 4 , 0x7f ) ; /* permission flags */
return ( True ) ;
}
/****************************************************************************
api_WPrintJobEnumerate
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static BOOL api_WPrintJobGetInfo ( connection_struct * conn , uint16 vuid , char * param , char * data ,
int mdrcnt , int mprcnt ,
char * * rdata , char * * rparam ,
int * rdata_len , int * rparam_len )
{
char * str1 = param + 2 ;
char * str2 = skip_string ( str1 , 1 ) ;
char * p = skip_string ( str2 , 1 ) ;
int uLevel ;
int count ;
int i ;
int snum ;
int job ;
struct pack_desc desc ;
print_queue_struct * queue = NULL ;
print_status_struct status ;
2001-03-23 06:12:58 +03:00
char * tmpdata = NULL ;
2000-04-19 11:00:22 +04:00
uLevel = SVAL ( p , 2 ) ;
memset ( ( char * ) & desc , ' \0 ' , sizeof ( desc ) ) ;
memset ( ( char * ) & status , ' \0 ' , sizeof ( status ) ) ;
DEBUG ( 3 , ( " WPrintJobGetInfo uLevel=%d uJobId=0x%X \n " , uLevel , SVAL ( p , 0 ) ) ) ;
/* check it's a supported varient */
if ( strcmp ( str1 , " WWrLh " ) ! = 0 ) return False ;
if ( ! check_printjob_info ( & desc , uLevel , str2 ) ) return False ;
job = SVAL ( p , 0 ) ;
2000-04-22 12:27:48 +04:00
snum = print_job_snum ( job ) ;
2000-04-19 11:00:22 +04:00
if ( snum < 0 | | ! VALID_SNUM ( snum ) ) return ( False ) ;
count = print_queue_status ( snum , & queue , & status ) ;
for ( i = 0 ; i < count ; i + + ) {
if ( queue [ i ] . job = = job ) break ;
}
2001-03-23 06:12:58 +03:00
if ( mdrcnt > 0 ) {
* rdata = REALLOC ( * rdata , mdrcnt ) ;
desc . base = * rdata ;
desc . buflen = mdrcnt ;
} else {
/*
* Don ' t return data but need to get correct length
* init_package will return wrong size if buflen = 0
*/
desc . buflen = getlen ( desc . format ) ;
desc . base = tmpdata = ( char * ) malloc ( desc . buflen ) ;
}
2000-04-19 11:00:22 +04:00
if ( init_package ( & desc , 1 , 0 ) ) {
if ( i < count ) {
fill_printjob_info ( conn , snum , uLevel , & desc , & queue [ i ] , i ) ;
* rdata_len = desc . usedlen ;
}
else {
desc . errcode = NERR_JobNotFound ;
* rdata_len = 0 ;
}
}
* rparam_len = 6 ;
* rparam = REALLOC ( * rparam , * rparam_len ) ;
SSVALS ( * rparam , 0 , desc . errcode ) ;
SSVAL ( * rparam , 2 , 0 ) ;
SSVAL ( * rparam , 4 , desc . neededlen ) ;
2001-09-17 15:25:41 +04:00
SAFE_FREE ( queue ) ;
SAFE_FREE ( tmpdata ) ;
2000-04-19 11:00:22 +04:00
DEBUG ( 4 , ( " WPrintJobGetInfo: errorcode %d \n " , desc . errcode ) ) ;
return ( True ) ;
}
static BOOL api_WPrintJobEnumerate ( connection_struct * conn , uint16 vuid , char * param , char * data ,
int mdrcnt , int mprcnt ,
char * * rdata , char * * rparam ,
int * rdata_len , int * rparam_len )
{
char * str1 = param + 2 ;
char * str2 = skip_string ( str1 , 1 ) ;
char * p = skip_string ( str2 , 1 ) ;
char * name = p ;
int uLevel ;
int count ;
int i , succnt = 0 ;
int snum ;
struct pack_desc desc ;
print_queue_struct * queue = NULL ;
print_status_struct status ;
memset ( ( char * ) & desc , ' \0 ' , sizeof ( desc ) ) ;
memset ( ( char * ) & status , ' \0 ' , sizeof ( status ) ) ;
p = skip_string ( p , 1 ) ;
uLevel = SVAL ( p , 0 ) ;
DEBUG ( 3 , ( " WPrintJobEnumerate uLevel=%d name=%s \n " , uLevel , name ) ) ;
/* check it's a supported varient */
if ( strcmp ( str1 , " zWrLeh " ) ! = 0 ) return False ;
if ( uLevel > 2 ) return False ; /* defined only for uLevel 0,1,2 */
if ( ! check_printjob_info ( & desc , uLevel , str2 ) ) return False ;
snum = lp_servicenumber ( name ) ;
if ( snum < 0 & & pcap_printername_ok ( name , NULL ) ) {
int pnum = lp_servicenumber ( PRINTERS_NAME ) ;
if ( pnum > = 0 ) {
lp_add_printer ( name , pnum ) ;
snum = lp_servicenumber ( name ) ;
}
}
if ( snum < 0 | | ! VALID_SNUM ( snum ) ) return ( False ) ;
count = print_queue_status ( snum , & queue , & status ) ;
if ( mdrcnt > 0 ) * rdata = REALLOC ( * rdata , mdrcnt ) ;
desc . base = * rdata ;
desc . buflen = mdrcnt ;
if ( init_package ( & desc , count , 0 ) ) {
succnt = 0 ;
for ( i = 0 ; i < count ; i + + ) {
fill_printjob_info ( conn , snum , uLevel , & desc , & queue [ i ] , i ) ;
if ( desc . errcode = = NERR_Success ) succnt = i + 1 ;
}
}
* rdata_len = desc . usedlen ;
* rparam_len = 8 ;
* rparam = REALLOC ( * rparam , * rparam_len ) ;
SSVALS ( * rparam , 0 , desc . errcode ) ;
SSVAL ( * rparam , 2 , 0 ) ;
SSVAL ( * rparam , 4 , succnt ) ;
SSVAL ( * rparam , 6 , count ) ;
2001-09-17 15:25:41 +04:00
SAFE_FREE ( queue ) ;
2000-04-19 11:00:22 +04:00
DEBUG ( 4 , ( " WPrintJobEnumerate: errorcode %d \n " , desc . errcode ) ) ;
return ( True ) ;
}
static int check_printdest_info ( struct pack_desc * desc ,
int uLevel , char * id )
{
desc - > subformat = NULL ;
switch ( uLevel ) {
case 0 : desc - > format = " B9 " ; break ;
case 1 : desc - > format = " B9B21WWzW " ; break ;
case 2 : desc - > format = " z " ; break ;
case 3 : desc - > format = " zzzWWzzzWW " ; break ;
default : return False ;
}
if ( strcmp ( desc - > format , id ) ! = 0 ) return False ;
return True ;
}
static void fill_printdest_info ( connection_struct * conn , int snum , int uLevel ,
struct pack_desc * desc )
{
char buf [ 100 ] ;
strncpy ( buf , SERVICE ( snum ) , sizeof ( buf ) - 1 ) ;
buf [ sizeof ( buf ) - 1 ] = 0 ;
strupper ( buf ) ;
if ( uLevel < = 1 ) {
PACKS ( desc , " B9 " , buf ) ; /* szName */
if ( uLevel = = 1 ) {
PACKS ( desc , " B21 " , " " ) ; /* szUserName */
PACKI ( desc , " W " , 0 ) ; /* uJobId */
PACKI ( desc , " W " , 0 ) ; /* fsStatus */
PACKS ( desc , " z " , " " ) ; /* pszStatus */
PACKI ( desc , " W " , 0 ) ; /* time */
}
}
if ( uLevel = = 2 | | uLevel = = 3 ) {
PACKS ( desc , " z " , buf ) ; /* pszPrinterName */
if ( uLevel = = 3 ) {
PACKS ( desc , " z " , " " ) ; /* pszUserName */
PACKS ( desc , " z " , " " ) ; /* pszLogAddr */
PACKI ( desc , " W " , 0 ) ; /* uJobId */
PACKI ( desc , " W " , 0 ) ; /* fsStatus */
PACKS ( desc , " z " , " " ) ; /* pszStatus */
PACKS ( desc , " z " , " " ) ; /* pszComment */
PACKS ( desc , " z " , " NULL " ) ; /* pszDrivers */
PACKI ( desc , " W " , 0 ) ; /* time */
PACKI ( desc , " W " , 0 ) ; /* pad1 */
}
}
}
static BOOL api_WPrintDestGetInfo ( connection_struct * conn , uint16 vuid , char * param , char * data ,
int mdrcnt , int mprcnt ,
char * * rdata , char * * rparam ,
int * rdata_len , int * rparam_len )
{
char * str1 = param + 2 ;
char * str2 = skip_string ( str1 , 1 ) ;
char * p = skip_string ( str2 , 1 ) ;
char * PrinterName = p ;
int uLevel ;
struct pack_desc desc ;
int snum ;
2001-03-23 06:12:58 +03:00
char * tmpdata = NULL ;
2000-04-19 11:00:22 +04:00
memset ( ( char * ) & desc , ' \0 ' , sizeof ( desc ) ) ;
p = skip_string ( p , 1 ) ;
uLevel = SVAL ( p , 0 ) ;
DEBUG ( 3 , ( " WPrintDestGetInfo uLevel=%d PrinterName=%s \n " , uLevel , PrinterName ) ) ;
/* check it's a supported varient */
if ( strcmp ( str1 , " zWrLh " ) ! = 0 ) return False ;
if ( ! check_printdest_info ( & desc , uLevel , str2 ) ) return False ;
snum = lp_servicenumber ( PrinterName ) ;
if ( snum < 0 & & pcap_printername_ok ( PrinterName , NULL ) ) {
int pnum = lp_servicenumber ( PRINTERS_NAME ) ;
if ( pnum > = 0 ) {
lp_add_printer ( PrinterName , pnum ) ;
snum = lp_servicenumber ( PrinterName ) ;
}
}
if ( snum < 0 ) {
* rdata_len = 0 ;
desc . errcode = NERR_DestNotFound ;
desc . neededlen = 0 ;
}
else {
2001-03-23 06:12:58 +03:00
if ( mdrcnt > 0 ) {
* rdata = REALLOC ( * rdata , mdrcnt ) ;
desc . base = * rdata ;
desc . buflen = mdrcnt ;
} else {
/*
* Don ' t return data but need to get correct length
* init_package will return wrong size if buflen = 0
*/
desc . buflen = getlen ( desc . format ) ;
desc . base = tmpdata = ( char * ) malloc ( desc . buflen ) ;
}
2000-04-19 11:00:22 +04:00
if ( init_package ( & desc , 1 , 0 ) ) {
fill_printdest_info ( conn , snum , uLevel , & desc ) ;
}
* rdata_len = desc . usedlen ;
}
* rparam_len = 6 ;
* rparam = REALLOC ( * rparam , * rparam_len ) ;
SSVALS ( * rparam , 0 , desc . errcode ) ;
SSVAL ( * rparam , 2 , 0 ) ;
SSVAL ( * rparam , 4 , desc . neededlen ) ;
DEBUG ( 4 , ( " WPrintDestGetInfo: errorcode %d \n " , desc . errcode ) ) ;
2001-09-17 15:25:41 +04:00
SAFE_FREE ( tmpdata ) ;
2000-04-19 11:00:22 +04:00
return ( True ) ;
}
static BOOL api_WPrintDestEnum ( connection_struct * conn , uint16 vuid , char * param , char * data ,
int mdrcnt , int mprcnt ,
char * * rdata , char * * rparam ,
int * rdata_len , int * rparam_len )
{
char * str1 = param + 2 ;
char * str2 = skip_string ( str1 , 1 ) ;
char * p = skip_string ( str2 , 1 ) ;
int uLevel ;
int queuecnt ;
int i , n , succnt = 0 ;
struct pack_desc desc ;
int services = lp_numservices ( ) ;
memset ( ( char * ) & desc , ' \0 ' , sizeof ( desc ) ) ;
uLevel = SVAL ( p , 0 ) ;
DEBUG ( 3 , ( " WPrintDestEnum uLevel=%d \n " , uLevel ) ) ;
/* check it's a supported varient */
if ( strcmp ( str1 , " WrLeh " ) ! = 0 ) return False ;
if ( ! check_printdest_info ( & desc , uLevel , str2 ) ) return False ;
queuecnt = 0 ;
for ( i = 0 ; i < services ; i + + )
if ( lp_snum_ok ( i ) & & lp_print_ok ( i ) & & lp_browseable ( i ) )
queuecnt + + ;
if ( mdrcnt > 0 ) * rdata = REALLOC ( * rdata , mdrcnt ) ;
desc . base = * rdata ;
desc . buflen = mdrcnt ;
if ( init_package ( & desc , queuecnt , 0 ) ) {
succnt = 0 ;
n = 0 ;
for ( i = 0 ; i < services ; i + + ) {
if ( lp_snum_ok ( i ) & & lp_print_ok ( i ) & & lp_browseable ( i ) ) {
fill_printdest_info ( conn , i , uLevel , & desc ) ;
n + + ;
if ( desc . errcode = = NERR_Success ) succnt = n ;
}
}
}
* rdata_len = desc . usedlen ;
* rparam_len = 8 ;
* rparam = REALLOC ( * rparam , * rparam_len ) ;
SSVALS ( * rparam , 0 , desc . errcode ) ;
SSVAL ( * rparam , 2 , 0 ) ;
SSVAL ( * rparam , 4 , succnt ) ;
SSVAL ( * rparam , 6 , queuecnt ) ;
DEBUG ( 4 , ( " WPrintDestEnumerate: errorcode %d \n " , desc . errcode ) ) ;
return ( True ) ;
}
static BOOL api_WPrintDriverEnum ( connection_struct * conn , uint16 vuid , char * param , char * data ,
int mdrcnt , int mprcnt ,
char * * rdata , char * * rparam ,
int * rdata_len , int * rparam_len )
{
char * str1 = param + 2 ;
char * str2 = skip_string ( str1 , 1 ) ;
char * p = skip_string ( str2 , 1 ) ;
int uLevel ;
int succnt ;
struct pack_desc desc ;
memset ( ( char * ) & desc , ' \0 ' , sizeof ( desc ) ) ;
uLevel = SVAL ( p , 0 ) ;
DEBUG ( 3 , ( " WPrintDriverEnum uLevel=%d \n " , uLevel ) ) ;
/* check it's a supported varient */
if ( strcmp ( str1 , " WrLeh " ) ! = 0 ) return False ;
if ( uLevel ! = 0 | | strcmp ( str2 , " B41 " ) ! = 0 ) return False ;
if ( mdrcnt > 0 ) * rdata = REALLOC ( * rdata , mdrcnt ) ;
desc . base = * rdata ;
desc . buflen = mdrcnt ;
if ( init_package ( & desc , 1 , 0 ) ) {
PACKS ( & desc , " B41 " , " NULL " ) ;
}
succnt = ( desc . errcode = = NERR_Success ? 1 : 0 ) ;
* rdata_len = desc . usedlen ;
* rparam_len = 8 ;
* rparam = REALLOC ( * rparam , * rparam_len ) ;
SSVALS ( * rparam , 0 , desc . errcode ) ;
SSVAL ( * rparam , 2 , 0 ) ;
SSVAL ( * rparam , 4 , succnt ) ;
SSVAL ( * rparam , 6 , 1 ) ;
DEBUG ( 4 , ( " WPrintDriverEnum: errorcode %d \n " , desc . errcode ) ) ;
return ( True ) ;
}
static BOOL api_WPrintQProcEnum ( connection_struct * conn , uint16 vuid , char * param , char * data ,
int mdrcnt , int mprcnt ,
char * * rdata , char * * rparam ,
int * rdata_len , int * rparam_len )
{
char * str1 = param + 2 ;
char * str2 = skip_string ( str1 , 1 ) ;
char * p = skip_string ( str2 , 1 ) ;
int uLevel ;
int succnt ;
struct pack_desc desc ;
memset ( ( char * ) & desc , ' \0 ' , sizeof ( desc ) ) ;
uLevel = SVAL ( p , 0 ) ;
DEBUG ( 3 , ( " WPrintQProcEnum uLevel=%d \n " , uLevel ) ) ;
/* check it's a supported varient */
if ( strcmp ( str1 , " WrLeh " ) ! = 0 ) return False ;
if ( uLevel ! = 0 | | strcmp ( str2 , " B13 " ) ! = 0 ) return False ;
if ( mdrcnt > 0 ) * rdata = REALLOC ( * rdata , mdrcnt ) ;
desc . base = * rdata ;
desc . buflen = mdrcnt ;
desc . format = str2 ;
if ( init_package ( & desc , 1 , 0 ) ) {
PACKS ( & desc , " B13 " , " lpd " ) ;
}
succnt = ( desc . errcode = = NERR_Success ? 1 : 0 ) ;
* rdata_len = desc . usedlen ;
* rparam_len = 8 ;
* rparam = REALLOC ( * rparam , * rparam_len ) ;
SSVALS ( * rparam , 0 , desc . errcode ) ;
SSVAL ( * rparam , 2 , 0 ) ;
SSVAL ( * rparam , 4 , succnt ) ;
SSVAL ( * rparam , 6 , 1 ) ;
DEBUG ( 4 , ( " WPrintQProcEnum: errorcode %d \n " , desc . errcode ) ) ;
return ( True ) ;
}
static BOOL api_WPrintPortEnum ( connection_struct * conn , uint16 vuid , char * param , char * data ,
int mdrcnt , int mprcnt ,
char * * rdata , char * * rparam ,
int * rdata_len , int * rparam_len )
{
char * str1 = param + 2 ;
char * str2 = skip_string ( str1 , 1 ) ;
char * p = skip_string ( str2 , 1 ) ;
int uLevel ;
int succnt ;
struct pack_desc desc ;
memset ( ( char * ) & desc , ' \0 ' , sizeof ( desc ) ) ;
uLevel = SVAL ( p , 0 ) ;
DEBUG ( 3 , ( " WPrintPortEnum uLevel=%d \n " , uLevel ) ) ;
/* check it's a supported varient */
if ( strcmp ( str1 , " WrLeh " ) ! = 0 ) return False ;
if ( uLevel ! = 0 | | strcmp ( str2 , " B9 " ) ! = 0 ) return False ;
if ( mdrcnt > 0 ) * rdata = REALLOC ( * rdata , mdrcnt ) ;
memset ( ( char * ) & desc , ' \0 ' , sizeof ( desc ) ) ;
desc . base = * rdata ;
desc . buflen = mdrcnt ;
desc . format = str2 ;
if ( init_package ( & desc , 1 , 0 ) ) {
PACKS ( & desc , " B13 " , " lp0 " ) ;
}
succnt = ( desc . errcode = = NERR_Success ? 1 : 0 ) ;
* rdata_len = desc . usedlen ;
* rparam_len = 8 ;
* rparam = REALLOC ( * rparam , * rparam_len ) ;
SSVALS ( * rparam , 0 , desc . errcode ) ;
SSVAL ( * rparam , 2 , 0 ) ;
SSVAL ( * rparam , 4 , succnt ) ;
SSVAL ( * rparam , 6 , 1 ) ;
DEBUG ( 4 , ( " WPrintPortEnum: errorcode %d \n " , desc . errcode ) ) ;
return ( True ) ;
}
/****************************************************************************
The buffer was too small
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static BOOL api_TooSmall ( connection_struct * conn , uint16 vuid , char * param , char * data ,
int mdrcnt , int mprcnt ,
char * * rdata , char * * rparam ,
int * rdata_len , int * rparam_len )
{
* rparam_len = MIN ( * rparam_len , mprcnt ) ;
* rparam = REALLOC ( * rparam , * rparam_len ) ;
* rdata_len = 0 ;
SSVAL ( * rparam , 0 , NERR_BufTooSmall ) ;
DEBUG ( 3 , ( " Supplied buffer too small in API command \n " ) ) ;
return ( True ) ;
}
/****************************************************************************
The request is not supported
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static BOOL api_Unsupported ( connection_struct * conn , uint16 vuid , char * param , char * data ,
int mdrcnt , int mprcnt ,
char * * rdata , char * * rparam ,
int * rdata_len , int * rparam_len )
{
* rparam_len = 4 ;
* rparam = REALLOC ( * rparam , * rparam_len ) ;
* rdata_len = 0 ;
SSVAL ( * rparam , 0 , NERR_notsupported ) ;
SSVAL ( * rparam , 2 , 0 ) ; /* converter word */
DEBUG ( 3 , ( " Unsupported API command \n " ) ) ;
return ( True ) ;
}
struct
{
char * name ;
int id ;
BOOL ( * fn ) ( connection_struct * , uint16 , char * , char * ,
int , int , char * * , char * * , int * , int * ) ;
int flags ;
} api_commands [ ] = {
{ " RNetShareEnum " , 0 , api_RNetShareEnum , 0 } ,
{ " RNetShareGetInfo " , 1 , api_RNetShareGetInfo , 0 } ,
{ " RNetServerGetInfo " , 13 , api_RNetServerGetInfo , 0 } ,
2001-08-28 20:05:55 +04:00
{ " RNetGroupEnum " , 47 , api_RNetGroupEnum , 0 } ,
2000-04-19 11:00:22 +04:00
{ " RNetGroupGetUsers " , 52 , api_RNetGroupGetUsers , 0 } ,
2001-08-28 20:05:55 +04:00
{ " RNetUserEnum " , 53 , api_RNetUserEnum , 0 } ,
2000-04-19 11:00:22 +04:00
{ " RNetUserGetInfo " , 56 , api_RNetUserGetInfo , 0 } ,
{ " NetUserGetGroups " , 59 , api_NetUserGetGroups , 0 } ,
{ " NetWkstaGetInfo " , 63 , api_NetWkstaGetInfo , 0 } ,
{ " DosPrintQEnum " , 69 , api_DosPrintQEnum , 0 } ,
{ " DosPrintQGetInfo " , 70 , api_DosPrintQGetInfo , 0 } ,
2000-10-18 10:22:20 +04:00
{ " WPrintQueuePause " , 74 , api_WPrintQueueCtrl , 0 } ,
{ " WPrintQueueResume " , 75 , api_WPrintQueueCtrl , 0 } ,
2000-04-19 11:00:22 +04:00
{ " WPrintJobEnumerate " , 76 , api_WPrintJobEnumerate , 0 } ,
{ " WPrintJobGetInfo " , 77 , api_WPrintJobGetInfo , 0 } ,
{ " RDosPrintJobDel " , 81 , api_RDosPrintJobDel , 0 } ,
{ " RDosPrintJobPause " , 82 , api_RDosPrintJobDel , 0 } ,
{ " RDosPrintJobResume " , 83 , api_RDosPrintJobDel , 0 } ,
{ " WPrintDestEnum " , 84 , api_WPrintDestEnum , 0 } ,
{ " WPrintDestGetInfo " , 85 , api_WPrintDestGetInfo , 0 } ,
{ " NetRemoteTOD " , 91 , api_NetRemoteTOD , 0 } ,
2000-10-18 10:22:20 +04:00
{ " WPrintQueuePurge " , 103 , api_WPrintQueueCtrl , 0 } ,
2000-04-19 11:00:22 +04:00
{ " NetServerEnum " , 104 , api_RNetServerEnum , 0 } ,
{ " WAccessGetUserPerms " , 105 , api_WAccessGetUserPerms , 0 } ,
{ " SetUserPassword " , 115 , api_SetUserPassword , 0 } ,
{ " WWkstaUserLogon " , 132 , api_WWkstaUserLogon , 0 } ,
{ " PrintJobInfo " , 147 , api_PrintJobInfo , 0 } ,
{ " WPrintDriverEnum " , 205 , api_WPrintDriverEnum , 0 } ,
{ " WPrintQProcEnum " , 206 , api_WPrintQProcEnum , 0 } ,
{ " WPrintPortEnum " , 207 , api_WPrintPortEnum , 0 } ,
{ " SamOEMChangePassword " , 214 , api_SamOEMChangePassword , 0 } ,
{ NULL , - 1 , api_Unsupported , 0 } } ;
/****************************************************************************
Handle remote api calls
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
int api_reply ( connection_struct * conn , uint16 vuid , char * outbuf , char * data , char * params ,
int tdscnt , int tpscnt , int mdrcnt , int mprcnt )
{
int api_command ;
char * rdata = NULL ;
char * rparam = NULL ;
int rdata_len = 0 ;
int rparam_len = 0 ;
BOOL reply = False ;
int i ;
if ( ! params ) {
DEBUG ( 0 , ( " ERROR: NULL params in api_reply() \n " ) ) ;
return 0 ;
}
api_command = SVAL ( params , 0 ) ;
DEBUG ( 3 , ( " Got API command %d of form <%s> <%s> (tdscnt=%d,tpscnt=%d,mdrcnt=%d,mprcnt=%d) \n " ,
api_command ,
params + 2 ,
skip_string ( params + 2 , 1 ) ,
tdscnt , tpscnt , mdrcnt , mprcnt ) ) ;
for ( i = 0 ; api_commands [ i ] . name ; i + + ) {
if ( api_commands [ i ] . id = = api_command & & api_commands [ i ] . fn ) {
DEBUG ( 3 , ( " Doing %s \n " , api_commands [ i ] . name ) ) ;
break ;
}
}
rdata = ( char * ) malloc ( 1024 ) ;
if ( rdata )
memset ( rdata , ' \0 ' , 1024 ) ;
rparam = ( char * ) malloc ( 1024 ) ;
if ( rparam )
memset ( rparam , ' \0 ' , 1024 ) ;
if ( ! rdata | | ! rparam ) {
DEBUG ( 0 , ( " api_reply: malloc fail ! \n " ) ) ;
return - 1 ;
}
reply = api_commands [ i ] . fn ( conn , vuid , params , data , mdrcnt , mprcnt ,
& rdata , & rparam , & rdata_len , & rparam_len ) ;
if ( rdata_len > mdrcnt | |
rparam_len > mprcnt ) {
reply = api_TooSmall ( conn , vuid , params , data , mdrcnt , mprcnt ,
& rdata , & rparam , & rdata_len , & rparam_len ) ;
}
/* if we get False back then it's actually unsupported */
if ( ! reply )
api_Unsupported ( conn , vuid , params , data , mdrcnt , mprcnt ,
& rdata , & rparam , & rdata_len , & rparam_len ) ;
send_trans_reply ( outbuf , rparam , rparam_len , rdata , rdata_len , False ) ;
2001-09-17 15:25:41 +04:00
SAFE_FREE ( rdata ) ;
SAFE_FREE ( rparam ) ;
2000-04-19 11:00:22 +04:00
return - 1 ;
}