2003-08-13 01:53:07 +00:00
/*
Unix SMB / CIFS implementation .
client directory search routines
Copyright ( C ) James Myers 2003 < myersjj @ samba . org >
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 .
*/
# include "includes.h"
/****************************************************************************
Old style search backend - process output .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static void smb_raw_search_backend ( struct cli_request * req ,
TALLOC_CTX * mem_ctx ,
uint16 count ,
void * private ,
BOOL ( * callback ) ( void * private , union smb_search_data * file ) )
{
union smb_search_data search_data ;
int i ;
char * p ;
if ( req - > in . data_size < 3 + count * 43 ) {
req - > status = NT_STATUS_INVALID_PARAMETER ;
return ;
}
p = req - > in . data + 3 ;
for ( i = 0 ; i < count ; i + + ) {
search_data . search . search_id = cli_req_pull_blob ( req , mem_ctx , p , 21 ) ;
search_data . search . attrib = CVAL ( p , 21 ) ;
search_data . search . write_time = make_unix_date ( p + 22 ) ;
search_data . search . size = IVAL ( p , 26 ) ;
cli_req_pull_ascii ( req , mem_ctx , & search_data . search . name , p + 30 , 13 , STR_ASCII ) ;
if ( ! callback ( private , & search_data ) ) {
break ;
}
p + = 43 ;
}
}
/****************************************************************************
Old style search first .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static NTSTATUS smb_raw_search_first_old ( struct cli_tree * tree ,
TALLOC_CTX * mem_ctx ,
union smb_search_first * io , void * private ,
BOOL ( * callback ) ( void * private , union smb_search_data * file ) )
{
struct cli_request * req ;
req = cli_request_setup ( tree , SMBsearch , 2 , 0 ) ;
if ( ! req ) {
return NT_STATUS_NO_MEMORY ;
}
SSVAL ( req - > out . vwv , VWV ( 0 ) , io - > search_first . in . max_count ) ;
SSVAL ( req - > out . vwv , VWV ( 1 ) , io - > search_first . in . search_attrib ) ;
cli_req_append_ascii4 ( req , io - > search_first . in . pattern , STR_TERMINATE ) ;
cli_req_append_var_block ( req , NULL , 0 ) ;
if ( ! cli_request_send ( req ) | |
! cli_request_receive ( req ) ) {
return cli_request_destroy ( req ) ;
}
if ( NT_STATUS_IS_OK ( req - > status ) ) {
io - > search_first . out . count = SVAL ( req - > in . vwv , VWV ( 0 ) ) ;
smb_raw_search_backend ( req , mem_ctx , io - > search_first . out . count , private , callback ) ;
}
return cli_request_destroy ( req ) ;
}
/****************************************************************************
Old style search next .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static NTSTATUS smb_raw_search_next_old ( struct cli_tree * tree ,
TALLOC_CTX * mem_ctx ,
union smb_search_next * io , void * private ,
BOOL ( * callback ) ( void * private , union smb_search_data * file ) )
{
struct cli_request * req ;
req = cli_request_setup ( tree , SMBsearch , 2 , 0 ) ;
if ( ! req ) {
return NT_STATUS_NO_MEMORY ;
}
SSVAL ( req - > out . vwv , VWV ( 0 ) , io - > search_next . in . max_count ) ;
SSVAL ( req - > out . vwv , VWV ( 1 ) , io - > search_next . in . search_attrib ) ;
cli_req_append_ascii4 ( req , " " , STR_TERMINATE ) ;
cli_req_append_var_block ( req , io - > search_next . in . search_id . data , 21 ) ;
if ( ! cli_request_send ( req ) | |
! cli_request_receive ( req ) ) {
return cli_request_destroy ( req ) ;
}
if ( NT_STATUS_IS_OK ( req - > status ) ) {
io - > search_next . out . count = SVAL ( req - > in . vwv , VWV ( 0 ) ) ;
smb_raw_search_backend ( req , mem_ctx , io - > search_next . out . count , private , callback ) ;
}
return cli_request_destroy ( req ) ;
}
/****************************************************************************
Very raw search first - returns param / data blobs .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static NTSTATUS smb_raw_search_first_blob ( struct cli_tree * tree ,
TALLOC_CTX * mem_ctx , /* used to allocate output blobs */
union smb_search_first * io ,
uint16 info_level ,
DATA_BLOB * out_param_blob ,
DATA_BLOB * out_data_blob )
{
struct smb_trans2 tp ;
uint16 setup = TRANSACT2_FINDFIRST ;
NTSTATUS status ;
tp . in . max_setup = 0 ;
tp . in . flags = 0 ;
tp . in . timeout = 0 ;
tp . in . setup_count = 1 ;
tp . in . data = data_blob ( NULL , 0 ) ;
tp . in . max_param = 1024 ;
tp . in . max_data = 8192 ;
tp . in . setup = & setup ;
tp . in . params = data_blob_talloc ( mem_ctx , NULL , 12 ) ;
if ( ! tp . in . params . data ) {
return NT_STATUS_NO_MEMORY ;
}
SSVAL ( tp . in . params . data , 0 , io - > t2ffirst . in . search_attrib ) ;
SSVAL ( tp . in . params . data , 2 , io - > t2ffirst . in . max_count ) ;
SSVAL ( tp . in . params . data , 4 , io - > t2ffirst . in . flags ) ;
SSVAL ( tp . in . params . data , 6 , info_level ) ;
SIVAL ( tp . in . params . data , 8 , io - > t2ffirst . in . storage_type ) ;
cli_blob_append_string ( tree - > session , mem_ctx , & tp . in . params ,
io - > t2ffirst . in . pattern , STR_TERMINATE ) ;
status = smb_raw_trans2 ( tree , mem_ctx , & tp ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
out_param_blob - > length = tp . out . params . length ;
out_param_blob - > data = tp . out . params . data ;
out_data_blob - > length = tp . out . data . length ;
out_data_blob - > data = tp . out . data . data ;
return NT_STATUS_OK ;
}
/****************************************************************************
Very raw search first - returns param / data blobs .
Used in CIFS - on - CIFS NTVFS .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static NTSTATUS smb_raw_search_next_blob ( struct cli_tree * tree ,
TALLOC_CTX * mem_ctx ,
union smb_search_next * io ,
uint16 info_level ,
DATA_BLOB * out_param_blob ,
DATA_BLOB * out_data_blob )
{
struct smb_trans2 tp ;
uint16 setup = TRANSACT2_FINDNEXT ;
NTSTATUS status ;
tp . in . max_setup = 0 ;
tp . in . flags = 0 ;
tp . in . timeout = 0 ;
tp . in . setup_count = 1 ;
tp . in . data = data_blob ( NULL , 0 ) ;
tp . in . max_param = 1024 ;
tp . in . max_data = 8192 ;
tp . in . setup = & setup ;
tp . in . params = data_blob_talloc ( mem_ctx , NULL , 12 ) ;
if ( ! tp . in . params . data ) {
return NT_STATUS_NO_MEMORY ;
}
SSVAL ( tp . in . params . data , 0 , io - > t2fnext . in . handle ) ;
SSVAL ( tp . in . params . data , 2 , io - > t2fnext . in . max_count ) ;
SSVAL ( tp . in . params . data , 4 , info_level ) ;
SIVAL ( tp . in . params . data , 6 , io - > t2fnext . in . resume_key ) ;
SSVAL ( tp . in . params . data , 10 , io - > t2fnext . in . flags ) ;
cli_blob_append_string ( tree - > session , mem_ctx , & tp . in . params ,
io - > t2fnext . in . last_name ,
STR_TERMINATE ) ;
status = smb_raw_trans2 ( tree , mem_ctx , & tp ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
out_param_blob - > length = tp . out . params . length ;
out_param_blob - > data = tp . out . params . data ;
out_data_blob - > length = tp . out . data . length ;
out_data_blob - > data = tp . out . data . data ;
return NT_STATUS_OK ;
}
/*
parse a trans2 search response .
Return the number of bytes consumed
return 0 for success with end of list
return - 1 for a parse error
*/
static int parse_trans2_search ( struct cli_tree * tree ,
TALLOC_CTX * mem_ctx ,
enum search_level level ,
uint16 flags ,
DATA_BLOB * blob ,
union smb_search_data * data )
{
uint_t len , ofs ;
switch ( level ) {
case RAW_SEARCH_GENERIC :
case RAW_SEARCH_SEARCH :
/* handled elsewhere */
return - 1 ;
case RAW_SEARCH_STANDARD :
if ( flags & FLAG_TRANS2_FIND_REQUIRE_RESUME ) {
if ( blob - > length < 4 ) return - 1 ;
data - > standard . resume_key = IVAL ( blob - > data , 0 ) ;
blob - > data + = 4 ;
blob - > length - = 4 ;
}
if ( blob - > length < 24 ) return - 1 ;
data - > standard . create_time = make_unix_date2 ( blob - > data + 0 ) ;
data - > standard . access_time = make_unix_date2 ( blob - > data + 4 ) ;
data - > standard . write_time = make_unix_date2 ( blob - > data + 8 ) ;
data - > standard . size = IVAL ( blob - > data , 12 ) ;
data - > standard . alloc_size = IVAL ( blob - > data , 16 ) ;
data - > standard . attrib = SVAL ( blob - > data , 20 ) ;
len = cli_blob_pull_string ( tree - > session , mem_ctx , blob ,
& data - > standard . name ,
2003-08-31 03:16:52 +00:00
22 , 23 , STR_LEN8BIT | STR_TERMINATE | STR_LEN_NOTERM ) ;
return len + 23 ;
2003-08-13 01:53:07 +00:00
case RAW_SEARCH_EA_SIZE :
if ( flags & FLAG_TRANS2_FIND_REQUIRE_RESUME ) {
if ( blob - > length < 4 ) return - 1 ;
data - > ea_size . resume_key = IVAL ( blob - > data , 0 ) ;
blob - > data + = 4 ;
blob - > length - = 4 ;
}
if ( blob - > length < 28 ) return - 1 ;
data - > ea_size . create_time = make_unix_date2 ( blob - > data + 0 ) ;
data - > ea_size . access_time = make_unix_date2 ( blob - > data + 4 ) ;
data - > ea_size . write_time = make_unix_date2 ( blob - > data + 8 ) ;
data - > ea_size . size = IVAL ( blob - > data , 12 ) ;
data - > ea_size . alloc_size = IVAL ( blob - > data , 16 ) ;
data - > ea_size . attrib = SVAL ( blob - > data , 20 ) ;
data - > ea_size . ea_size = IVAL ( blob - > data , 22 ) ;
len = cli_blob_pull_string ( tree - > session , mem_ctx , blob ,
& data - > ea_size . name ,
2003-08-31 03:16:52 +00:00
26 , 27 , STR_LEN8BIT | STR_TERMINATE | STR_NOALIGN ) ;
return len + 27 + 1 ;
2003-08-13 01:53:07 +00:00
case RAW_SEARCH_DIRECTORY_INFO :
if ( blob - > length < 65 ) return - 1 ;
ofs = IVAL ( blob - > data , 0 ) ;
data - > directory_info . file_index = IVAL ( blob - > data , 4 ) ;
data - > directory_info . create_time = cli_pull_nttime ( blob - > data , 8 ) ;
data - > directory_info . access_time = cli_pull_nttime ( blob - > data , 16 ) ;
data - > directory_info . write_time = cli_pull_nttime ( blob - > data , 24 ) ;
data - > directory_info . change_time = cli_pull_nttime ( blob - > data , 32 ) ;
data - > directory_info . size = BVAL ( blob - > data , 40 ) ;
data - > directory_info . alloc_size = BVAL ( blob - > data , 48 ) ;
data - > directory_info . attrib = IVAL ( blob - > data , 56 ) ;
len = cli_blob_pull_string ( tree - > session , mem_ctx , blob ,
& data - > directory_info . name ,
60 , 64 , 0 ) ;
if ( ofs ! = 0 & & ofs < 64 + len ) {
return - 1 ;
}
return ofs ;
case RAW_SEARCH_FULL_DIRECTORY_INFO :
if ( blob - > length < 69 ) return - 1 ;
ofs = IVAL ( blob - > data , 0 ) ;
data - > full_directory_info . file_index = IVAL ( blob - > data , 4 ) ;
data - > full_directory_info . create_time = cli_pull_nttime ( blob - > data , 8 ) ;
data - > full_directory_info . access_time = cli_pull_nttime ( blob - > data , 16 ) ;
data - > full_directory_info . write_time = cli_pull_nttime ( blob - > data , 24 ) ;
data - > full_directory_info . change_time = cli_pull_nttime ( blob - > data , 32 ) ;
data - > full_directory_info . size = BVAL ( blob - > data , 40 ) ;
data - > full_directory_info . alloc_size = BVAL ( blob - > data , 48 ) ;
data - > full_directory_info . attrib = IVAL ( blob - > data , 56 ) ;
data - > full_directory_info . ea_size = IVAL ( blob - > data , 64 ) ;
len = cli_blob_pull_string ( tree - > session , mem_ctx , blob ,
& data - > full_directory_info . name ,
60 , 68 , 0 ) ;
if ( ofs ! = 0 & & ofs < 68 + len ) {
return - 1 ;
}
return ofs ;
case RAW_SEARCH_NAME_INFO :
if ( blob - > length < 13 ) return - 1 ;
ofs = IVAL ( blob - > data , 0 ) ;
data - > name_info . file_index = IVAL ( blob - > data , 4 ) ;
len = cli_blob_pull_string ( tree - > session , mem_ctx , blob ,
& data - > name_info . name ,
8 , 12 , 0 ) ;
if ( ofs ! = 0 & & ofs < 12 + len ) {
return - 1 ;
}
return ofs ;
case RAW_SEARCH_BOTH_DIRECTORY_INFO :
if ( blob - > length < 95 ) return - 1 ;
ofs = IVAL ( blob - > data , 0 ) ;
data - > both_directory_info . file_index = IVAL ( blob - > data , 4 ) ;
data - > both_directory_info . create_time = cli_pull_nttime ( blob - > data , 8 ) ;
data - > both_directory_info . access_time = cli_pull_nttime ( blob - > data , 16 ) ;
data - > both_directory_info . write_time = cli_pull_nttime ( blob - > data , 24 ) ;
data - > both_directory_info . change_time = cli_pull_nttime ( blob - > data , 32 ) ;
data - > both_directory_info . size = BVAL ( blob - > data , 40 ) ;
data - > both_directory_info . alloc_size = BVAL ( blob - > data , 48 ) ;
data - > both_directory_info . attrib = IVAL ( blob - > data , 56 ) ;
data - > both_directory_info . ea_size = IVAL ( blob - > data , 64 ) ;
cli_blob_pull_string ( tree - > session , mem_ctx , blob ,
& data - > both_directory_info . short_name ,
68 , 70 , STR_LEN8BIT | STR_UNICODE ) ;
len = cli_blob_pull_string ( tree - > session , mem_ctx , blob ,
& data - > both_directory_info . name ,
60 , 94 , 0 ) ;
if ( ofs ! = 0 & & ofs < 94 + len ) {
return - 1 ;
}
return ofs ;
case RAW_SEARCH_261 :
if ( blob - > length < 81 ) return - 1 ;
ofs = IVAL ( blob - > data , 0 ) ;
data - > level_261 . file_index = IVAL ( blob - > data , 4 ) ;
data - > level_261 . create_time = cli_pull_nttime ( blob - > data , 8 ) ;
data - > level_261 . access_time = cli_pull_nttime ( blob - > data , 16 ) ;
data - > level_261 . write_time = cli_pull_nttime ( blob - > data , 24 ) ;
data - > level_261 . change_time = cli_pull_nttime ( blob - > data , 32 ) ;
data - > level_261 . size = BVAL ( blob - > data , 40 ) ;
data - > level_261 . alloc_size = BVAL ( blob - > data , 48 ) ;
data - > level_261 . attrib = IVAL ( blob - > data , 56 ) ;
data - > level_261 . ea_size = IVAL ( blob - > data , 64 ) ;
data - > level_261 . unknown [ 0 ] = IVAL ( blob - > data , 68 ) ;
data - > level_261 . unknown [ 1 ] = IVAL ( blob - > data , 72 ) ;
data - > level_261 . unknown [ 2 ] = IVAL ( blob - > data , 76 ) ;
len = cli_blob_pull_string ( tree - > session , mem_ctx , blob ,
& data - > level_261 . name ,
60 , 80 , 0 ) ;
if ( ofs ! = 0 & & ofs < 80 + len ) {
return - 1 ;
}
return ofs ;
case RAW_SEARCH_262 :
if ( blob - > length < 105 ) return - 1 ;
ofs = IVAL ( blob - > data , 0 ) ;
data - > level_262 . file_index = IVAL ( blob - > data , 4 ) ;
data - > level_262 . create_time = cli_pull_nttime ( blob - > data , 8 ) ;
data - > level_262 . access_time = cli_pull_nttime ( blob - > data , 16 ) ;
data - > level_262 . write_time = cli_pull_nttime ( blob - > data , 24 ) ;
data - > level_262 . change_time = cli_pull_nttime ( blob - > data , 32 ) ;
data - > level_262 . size = BVAL ( blob - > data , 40 ) ;
data - > level_262 . alloc_size = BVAL ( blob - > data , 48 ) ;
data - > level_262 . attrib = SVAL ( blob - > data , 56 ) ;
data - > level_262 . ea_size = IVAL ( blob - > data , 64 ) ;
cli_blob_pull_string ( tree - > session , mem_ctx , blob ,
& data - > level_262 . short_name ,
68 , 70 , STR_LEN8BIT | STR_UNICODE ) ;
data - > level_262 . unknown [ 0 ] = IVAL ( blob - > data , 94 ) ;
data - > level_262 . unknown [ 1 ] = IVAL ( blob - > data , 98 ) ;
len = cli_blob_pull_string ( tree - > session , mem_ctx , blob ,
& data - > level_262 . name ,
60 , 104 , 0 ) ;
if ( ofs ! = 0 & & ofs < 104 + len ) {
return - 1 ;
}
return ofs ;
}
/* invalid level */
return - 1 ;
}
/****************************************************************************
Trans2 search backend - process output .
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
static NTSTATUS smb_raw_t2search_backend ( struct cli_tree * tree ,
TALLOC_CTX * mem_ctx ,
enum search_level level ,
uint16 flags ,
int16 count ,
DATA_BLOB * blob ,
void * private ,
BOOL ( * callback ) ( void * private , union smb_search_data * file ) )
{
int i ;
DATA_BLOB blob2 ;
blob2 . data = blob - > data ;
blob2 . length = blob - > length ;
for ( i = 0 ; i < count ; i + + ) {
union smb_search_data search_data ;
uint_t len ;
len = parse_trans2_search ( tree , mem_ctx , level , flags , & blob2 , & search_data ) ;
if ( len = = - 1 ) {
return NT_STATUS_INVALID_PARAMETER ;
}
/* the callback function can tell us that no more will
fit - in that case we stop , but it isn ' t an error */
if ( ! callback ( private , & search_data ) ) {
break ;
}
if ( len = = 0 ) break ;
blob2 . data + = len ;
blob2 . length - = len ;
}
return NT_STATUS_OK ;
}
/* Implements trans2findfirst2 and old search
*/
NTSTATUS smb_raw_search_first ( struct cli_tree * tree ,
TALLOC_CTX * mem_ctx ,
union smb_search_first * io , void * private ,
BOOL ( * callback ) ( void * private , union smb_search_data * file ) )
{
uint16 info_level = 0 ;
DATA_BLOB p_blob , d_blob ;
NTSTATUS status ;
if ( io - > generic . level = = RAW_SEARCH_SEARCH ) {
return smb_raw_search_first_old ( tree , mem_ctx , io , private , callback ) ;
}
if ( io - > generic . level > = RAW_SEARCH_GENERIC ) {
return NT_STATUS_INVALID_LEVEL ;
}
info_level = ( uint16 ) io - > generic . level ;
status = smb_raw_search_first_blob ( tree , mem_ctx ,
io , info_level , & p_blob , & d_blob ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
if ( p_blob . length ! = 10 ) {
DEBUG ( 1 , ( " smb_raw_search_first: parms wrong size %d != expected_param_size \n " ,
p_blob . length ) ) ;
return NT_STATUS_INVALID_PARAMETER ;
}
/* process output data */
io - > t2ffirst . out . handle = SVAL ( p_blob . data , 0 ) ;
io - > t2ffirst . out . count = SVAL ( p_blob . data , 2 ) ;
io - > t2ffirst . out . end_of_search = SVAL ( p_blob . data , 4 ) ;
status = smb_raw_t2search_backend ( tree , mem_ctx ,
io - > generic . level ,
io - > t2ffirst . in . flags , io - > t2ffirst . out . count ,
& d_blob , private , callback ) ;
return status ;
}
/* Implements trans2findnext2 and old smbsearch
*/
NTSTATUS smb_raw_search_next ( struct cli_tree * tree ,
TALLOC_CTX * mem_ctx ,
union smb_search_next * io , void * private ,
BOOL ( * callback ) ( void * private , union smb_search_data * file ) )
{
uint16 info_level = 0 ;
DATA_BLOB p_blob , d_blob ;
NTSTATUS status ;
if ( io - > generic . level = = RAW_SEARCH_SEARCH ) {
return smb_raw_search_next_old ( tree , mem_ctx , io , private , callback ) ;
}
if ( io - > generic . level > = RAW_SEARCH_GENERIC ) {
return NT_STATUS_INVALID_LEVEL ;
}
info_level = ( uint16 ) io - > generic . level ;
status = smb_raw_search_next_blob ( tree , mem_ctx ,
io , info_level , & p_blob , & d_blob ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
if ( p_blob . length ! = 8 ) {
DEBUG ( 1 , ( " smb_raw_search_next: parms wrong size %d != expected_param_size \n " ,
p_blob . length ) ) ;
return NT_STATUS_INVALID_PARAMETER ;
}
/* process output data */
io - > t2fnext . out . count = SVAL ( p_blob . data , 0 ) ;
io - > t2fnext . out . end_of_search = SVAL ( p_blob . data , 2 ) ;
status = smb_raw_t2search_backend ( tree , mem_ctx ,
io - > generic . level ,
io - > t2fnext . in . flags , io - > t2fnext . out . count ,
& d_blob , private , callback ) ;
return status ;
}
/*
Implements trans2findclose2
*/
NTSTATUS smb_raw_search_close ( struct cli_tree * tree ,
union smb_search_close * io )
{
struct cli_request * req ;
req = cli_request_setup ( tree , SMBfindclose , 1 , 0 ) ;
if ( ! req ) {
return NT_STATUS_NO_MEMORY ;
}
SSVAL ( req - > out . vwv , VWV ( 0 ) , io - > findclose . in . handle ) ;
if ( cli_request_send ( req ) ) {
cli_request_receive ( req ) ;
}
return cli_request_destroy ( req ) ;
}