2005-04-17 02:20:36 +04:00
/*
* fs / cifs_debug . c
*
2005-04-29 09:41:07 +04:00
* Copyright ( C ) International Business Machines Corp . , 2000 , 2005
2005-04-17 02:20:36 +04:00
*
* Modified by Steve French ( sfrench @ us . ibm . com )
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
2007-06-06 00:35:06 +04:00
* the Free Software Foundation ; either version 2 of the License , or
2005-04-17 02:20:36 +04:00
* ( at your option ) any later version .
2007-06-06 00:35:06 +04:00
*
2005-04-17 02:20:36 +04:00
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See
* the GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
2007-06-06 00:35:06 +04:00
* along with this program ; if not , write to the Free Software
2005-04-17 02:20:36 +04:00
* Foundation , Inc . , 59 Temple Place , Suite 330 , Boston , MA 02111 - 1307 USA
*/
# include <linux/fs.h>
# include <linux/string.h>
# include <linux/ctype.h>
# include <linux/module.h>
# include <linux/proc_fs.h>
# include <asm/uaccess.h>
# include "cifspdu.h"
# include "cifsglob.h"
# include "cifsproto.h"
# include "cifs_debug.h"
2005-04-29 09:41:06 +04:00
# include "cifsfs.h"
2005-04-17 02:20:36 +04:00
void
cifs_dump_mem ( char * label , void * data , int length )
{
int i , j ;
int * intptr = data ;
char * charptr = data ;
char buf [ 10 ] , line [ 80 ] ;
2007-06-06 00:35:06 +04:00
printk ( KERN_DEBUG " %s: dump of %d bytes of data at 0x%p \n " ,
2005-04-17 02:20:36 +04:00
label , length , data ) ;
for ( i = 0 ; i < length ; i + = 16 ) {
line [ 0 ] = 0 ;
for ( j = 0 ; ( j < 4 ) & & ( i + j * 4 < length ) ; j + + ) {
sprintf ( buf , " %08x " , intptr [ i / 4 + j ] ) ;
strcat ( line , buf ) ;
}
buf [ 0 ] = ' ' ;
buf [ 2 ] = 0 ;
for ( j = 0 ; ( j < 16 ) & & ( i + j < length ) ; j + + ) {
buf [ 1 ] = isprint ( charptr [ i + j ] ) ? charptr [ i + j ] : ' . ' ;
strcat ( line , buf ) ;
}
printk ( KERN_DEBUG " %s \n " , line ) ;
}
}
2006-06-01 02:40:51 +04:00
# ifdef CONFIG_CIFS_DEBUG2
2007-06-25 01:15:44 +04:00
void cifs_dump_detail ( struct smb_hdr * smb )
2006-06-01 02:40:51 +04:00
{
2007-06-06 00:35:06 +04:00
cERROR ( 1 , ( " Cmd: %d Err: 0x%x Flags: 0x%x Flgs2: 0x%x Mid: %d Pid: %d " ,
2006-06-01 02:40:51 +04:00
smb - > Command , smb - > Status . CifsError ,
smb - > Flags , smb - > Flags2 , smb - > Mid , smb - > Pid ) ) ;
2007-06-06 00:35:06 +04:00
cERROR ( 1 , ( " smb buf %p len %d " , smb , smbCalcSize_LE ( smb ) ) ) ;
2006-06-01 02:40:51 +04:00
}
2007-06-25 01:15:44 +04:00
void cifs_dump_mids ( struct TCP_Server_Info * server )
2006-06-01 02:40:51 +04:00
{
struct list_head * tmp ;
2007-06-25 01:15:44 +04:00
struct mid_q_entry * mid_entry ;
2006-06-01 02:40:51 +04:00
2007-06-06 00:35:06 +04:00
if ( server = = NULL )
2006-06-01 02:40:51 +04:00
return ;
2007-06-06 00:35:06 +04:00
cERROR ( 1 , ( " Dump pending requests: " ) ) ;
2006-06-01 02:40:51 +04:00
spin_lock ( & GlobalMid_Lock ) ;
list_for_each ( tmp , & server - > pending_mid_q ) {
mid_entry = list_entry ( tmp , struct mid_q_entry , qhead ) ;
2007-06-06 00:35:06 +04:00
if ( mid_entry ) {
cERROR ( 1 , ( " State: %d Cmd: %d Pid: %d Tsk: %p Mid %d " ,
2006-06-01 02:40:51 +04:00
mid_entry - > midState ,
( int ) mid_entry - > command ,
mid_entry - > pid ,
mid_entry - > tsk ,
mid_entry - > mid ) ) ;
# ifdef CONFIG_CIFS_STATS2
2007-06-06 00:35:06 +04:00
cERROR ( 1 , ( " IsLarge: %d buf: %p time rcv: %ld now: %ld " ,
2006-06-01 02:40:51 +04:00
mid_entry - > largeBuf ,
mid_entry - > resp_buf ,
mid_entry - > when_received ,
jiffies ) ) ;
# endif /* STATS2 */
2007-06-06 00:35:06 +04:00
cERROR ( 1 , ( " IsMult: %d IsEnd: %d " , mid_entry - > multiRsp ,
2006-06-01 02:40:51 +04:00
mid_entry - > multiEnd ) ) ;
2007-06-06 00:35:06 +04:00
if ( mid_entry - > resp_buf ) {
2006-06-01 02:40:51 +04:00
cifs_dump_detail ( mid_entry - > resp_buf ) ;
cifs_dump_mem ( " existing buf: " ,
2008-02-08 02:25:02 +03:00
mid_entry - > resp_buf , 62 ) ;
2006-06-01 02:40:51 +04:00
}
}
}
spin_unlock ( & GlobalMid_Lock ) ;
}
# endif /* CONFIG_CIFS_DEBUG2 */
2005-04-17 02:20:36 +04:00
# ifdef CONFIG_PROC_FS
static int
cifs_debug_data_read ( char * buf , char * * beginBuffer , off_t offset ,
int count , int * eof , void * data )
{
struct list_head * tmp ;
struct list_head * tmp1 ;
2007-06-25 01:15:44 +04:00
struct mid_q_entry * mid_entry ;
2005-04-17 02:20:36 +04:00
struct cifsSesInfo * ses ;
struct cifsTconInfo * tcon ;
int i ;
int length = 0 ;
2007-06-25 01:15:44 +04:00
char * original_buf = buf ;
2005-04-17 02:20:36 +04:00
* beginBuffer = buf + offset ;
length =
sprintf ( buf ,
" Display Internal CIFS Data Structures for Debugging \n "
" --------------------------------------------------- \n " ) ;
buf + = length ;
2007-06-06 00:35:06 +04:00
length = sprintf ( buf , " CIFS Version %s \n " , CIFS_VERSION ) ;
2005-04-29 09:41:06 +04:00
buf + = length ;
2007-06-06 00:35:06 +04:00
length = sprintf ( buf ,
" Active VFS Requests: %d \n " , GlobalTotalActiveXid ) ;
2005-10-07 20:51:05 +04:00
buf + = length ;
2005-04-29 09:41:06 +04:00
length = sprintf ( buf , " Servers: " ) ;
2005-04-17 02:20:36 +04:00
buf + = length ;
i = 0 ;
read_lock ( & GlobalSMBSeslock ) ;
list_for_each ( tmp , & GlobalSMBSessionList ) {
i + + ;
ses = list_entry ( tmp , struct cifsSesInfo , cifsSessionList ) ;
2007-06-06 00:35:06 +04:00
if ( ( ses - > serverDomain = = NULL ) | | ( ses - > serverOS = = NULL ) | |
2005-04-29 09:41:08 +04:00
( ses - > serverNOS = = NULL ) ) {
2007-01-22 02:19:01 +03:00
buf + = sprintf ( buf , " \n entry for %s not fully "
" displayed \n \t " , ses - > serverName ) ;
2005-04-29 09:41:08 +04:00
} else {
length =
sprintf ( buf ,
2007-06-06 00:35:06 +04:00
" \n %d) Name: %s Domain: %s Mounts: %d OS: "
" %s \n \t NOS: %s \t Capability: 0x%x \n \t SMB "
" session status: %d \t " ,
2005-04-29 09:41:07 +04:00
i , ses - > serverName , ses - > serverDomain ,
atomic_read ( & ses - > inUse ) ,
ses - > serverOS , ses - > serverNOS ,
2007-06-06 00:35:06 +04:00
ses - > capabilities , ses - > status ) ;
2005-04-29 09:41:08 +04:00
buf + = length ;
}
2007-06-06 00:35:06 +04:00
if ( ses - > server ) {
buf + = sprintf ( buf , " TCP status: %d \n \t Local Users To "
" Server: %d SecMode: 0x%x Req On Wire: %d " ,
2005-04-17 02:20:36 +04:00
ses - > server - > tcpStatus ,
atomic_read ( & ses - > server - > socketUseCount ) ,
ses - > server - > secMode ,
atomic_read ( & ses - > server - > inFlight ) ) ;
2005-10-07 20:51:05 +04:00
# ifdef CONFIG_CIFS_STATS2
2005-10-10 21:34:22 +04:00
buf + = sprintf ( buf , " In Send: %d In MaxReq Wait: %d " ,
2007-06-06 00:35:06 +04:00
atomic_read ( & ses - > server - > inSend ) ,
2005-10-07 20:51:05 +04:00
atomic_read ( & ses - > server - > num_waiters ) ) ;
# endif
2005-04-29 09:41:06 +04:00
length = sprintf ( buf , " \n MIDs: \n " ) ;
2005-04-17 02:20:36 +04:00
buf + = length ;
spin_lock ( & GlobalMid_Lock ) ;
list_for_each ( tmp1 , & ses - > server - > pending_mid_q ) {
mid_entry = list_entry ( tmp1 , struct
mid_q_entry ,
qhead ) ;
2007-06-06 00:35:06 +04:00
if ( mid_entry ) {
length = sprintf ( buf ,
" State: %d com: %d pid: "
" %d tsk: %p mid %d \n " ,
mid_entry - > midState ,
( int ) mid_entry - > command ,
mid_entry - > pid ,
mid_entry - > tsk ,
mid_entry - > mid ) ;
2005-04-17 02:20:36 +04:00
buf + = length ;
}
}
2007-06-06 00:35:06 +04:00
spin_unlock ( & GlobalMid_Lock ) ;
2005-04-17 02:20:36 +04:00
}
}
read_unlock ( & GlobalSMBSeslock ) ;
sprintf ( buf , " \n " ) ;
buf + + ;
2005-04-29 09:41:06 +04:00
length = sprintf ( buf , " Shares: " ) ;
2005-04-17 02:20:36 +04:00
buf + = length ;
i = 0 ;
read_lock ( & GlobalSMBSeslock ) ;
list_for_each ( tmp , & GlobalTreeConnectionList ) {
__u32 dev_type ;
i + + ;
tcon = list_entry ( tmp , struct cifsTconInfo , cifsConnectionList ) ;
dev_type = le32_to_cpu ( tcon - > fsDevInfo . DeviceType ) ;
2007-09-15 07:01:17 +04:00
length = sprintf ( buf , " \n %d) %s Uses: %d " , i ,
tcon - > treeName , atomic_read ( & tcon - > useCount ) ) ;
buf + = length ;
if ( tcon - > nativeFileSystem ) {
2007-09-15 07:43:47 +04:00
length = sprintf ( buf , " Type: %s " ,
tcon - > nativeFileSystem ) ;
2007-09-15 07:01:17 +04:00
buf + = length ;
}
length = sprintf ( buf , " DevInfo: 0x%x Attributes: 0x%x "
" \n PathComponentMax: %d Status: %d " ,
2005-04-17 02:20:36 +04:00
le32_to_cpu ( tcon - > fsDevInfo . DeviceCharacteristics ) ,
le32_to_cpu ( tcon - > fsAttrInfo . Attributes ) ,
le32_to_cpu ( tcon - > fsAttrInfo . MaxPathNameComponentLength ) ,
tcon - > tidStatus ) ;
2007-06-06 00:35:06 +04:00
buf + = length ;
2005-04-17 02:20:36 +04:00
if ( dev_type = = FILE_DEVICE_DISK )
length = sprintf ( buf , " type: DISK " ) ;
else if ( dev_type = = FILE_DEVICE_CD_ROM )
length = sprintf ( buf , " type: CDROM " ) ;
else
length =
sprintf ( buf , " type: %d " , dev_type ) ;
buf + = length ;
2007-06-06 00:35:06 +04:00
if ( tcon - > tidStatus = = CifsNeedReconnect ) {
2005-04-17 02:20:36 +04:00
buf + = sprintf ( buf , " \t DISCONNECTED " ) ;
length + = 14 ;
}
}
read_unlock ( & GlobalSMBSeslock ) ;
length = sprintf ( buf , " \n " ) ;
buf + = length ;
/* BB add code to dump additional info such as TCP session info now */
/* Now calculate total size of returned data */
length = buf - original_buf ;
2007-06-06 00:35:06 +04:00
if ( offset + count > = length )
2005-04-17 02:20:36 +04:00
* eof = 1 ;
2007-06-06 00:35:06 +04:00
if ( length < offset ) {
2005-04-17 02:20:36 +04:00
* eof = 1 ;
return 0 ;
} else {
length = length - offset ;
}
if ( length > count )
length = count ;
return length ;
}
# ifdef CONFIG_CIFS_STATS
2005-10-12 06:58:06 +04:00
static int
cifs_stats_write ( struct file * file , const char __user * buffer ,
2007-06-06 00:35:06 +04:00
unsigned long count , void * data )
2005-10-12 06:58:06 +04:00
{
2007-06-06 00:35:06 +04:00
char c ;
int rc ;
2005-10-12 06:58:06 +04:00
struct list_head * tmp ;
struct cifsTconInfo * tcon ;
2007-06-06 00:35:06 +04:00
rc = get_user ( c , buffer ) ;
if ( rc )
return rc ;
2005-10-12 06:58:06 +04:00
2007-06-06 00:35:06 +04:00
if ( c = = ' 1 ' | | c = = ' y ' | | c = = ' Y ' | | c = = ' 0 ' ) {
2005-10-12 06:58:06 +04:00
read_lock ( & GlobalSMBSeslock ) ;
2005-12-04 00:58:57 +03:00
# ifdef CONFIG_CIFS_STATS2
atomic_set ( & totBufAllocCount , 0 ) ;
atomic_set ( & totSmBufAllocCount , 0 ) ;
# endif /* CONFIG_CIFS_STATS2 */
2005-10-12 06:58:06 +04:00
list_for_each ( tmp , & GlobalTreeConnectionList ) {
tcon = list_entry ( tmp , struct cifsTconInfo ,
cifsConnectionList ) ;
atomic_set ( & tcon - > num_smbs_sent , 0 ) ;
atomic_set ( & tcon - > num_writes , 0 ) ;
atomic_set ( & tcon - > num_reads , 0 ) ;
atomic_set ( & tcon - > num_oplock_brks , 0 ) ;
atomic_set ( & tcon - > num_opens , 0 ) ;
atomic_set ( & tcon - > num_closes , 0 ) ;
atomic_set ( & tcon - > num_deletes , 0 ) ;
atomic_set ( & tcon - > num_mkdirs , 0 ) ;
atomic_set ( & tcon - > num_rmdirs , 0 ) ;
atomic_set ( & tcon - > num_renames , 0 ) ;
atomic_set ( & tcon - > num_t2renames , 0 ) ;
atomic_set ( & tcon - > num_ffirst , 0 ) ;
atomic_set ( & tcon - > num_fnext , 0 ) ;
atomic_set ( & tcon - > num_fclose , 0 ) ;
atomic_set ( & tcon - > num_hardlinks , 0 ) ;
atomic_set ( & tcon - > num_symlinks , 0 ) ;
atomic_set ( & tcon - > num_locks , 0 ) ;
}
read_unlock ( & GlobalSMBSeslock ) ;
}
2007-06-06 00:35:06 +04:00
return count ;
2005-10-12 06:58:06 +04:00
}
2005-04-17 02:20:36 +04:00
static int
cifs_stats_read ( char * buf , char * * beginBuffer , off_t offset ,
int count , int * eof , void * data )
{
2007-06-06 00:35:06 +04:00
int item_length , i , length ;
2005-04-17 02:20:36 +04:00
struct list_head * tmp ;
struct cifsTconInfo * tcon ;
* beginBuffer = buf + offset ;
length = sprintf ( buf ,
" Resources in use \n CIFS Session: %d \n " ,
sesInfoAllocCount . counter ) ;
buf + = length ;
2007-06-06 00:35:06 +04:00
item_length =
sprintf ( buf , " Share (unique mount targets): %d \n " ,
2005-04-17 02:20:36 +04:00
tconInfoAllocCount . counter ) ;
length + = item_length ;
2007-06-06 00:35:06 +04:00
buf + = item_length ;
item_length =
sprintf ( buf , " SMB Request/Response Buffer: %d Pool size: %d \n " ,
2005-04-29 09:41:07 +04:00
bufAllocCount . counter ,
cifs_min_rcv + tcpSesAllocCount . counter ) ;
2005-04-17 02:20:36 +04:00
length + = item_length ;
buf + = item_length ;
2007-06-06 00:35:06 +04:00
item_length =
sprintf ( buf , " SMB Small Req/Resp Buffer: %d Pool size: %d \n " ,
smBufAllocCount . counter , cifs_min_small ) ;
2005-04-17 02:20:36 +04:00
length + = item_length ;
buf + = item_length ;
2005-12-04 01:11:37 +03:00
# ifdef CONFIG_CIFS_STATS2
2007-06-06 00:35:06 +04:00
item_length = sprintf ( buf , " Total Large %d Small %d Allocations \n " ,
2005-12-04 01:11:37 +03:00
atomic_read ( & totBufAllocCount ) ,
2007-06-06 00:35:06 +04:00
atomic_read ( & totSmBufAllocCount ) ) ;
2005-12-04 01:11:37 +03:00
length + = item_length ;
buf + = item_length ;
# endif /* CONFIG_CIFS_STATS2 */
2007-06-06 00:35:06 +04:00
item_length =
sprintf ( buf , " Operations (MIDs): %d \n " ,
2005-04-17 02:20:36 +04:00
midCount . counter ) ;
length + = item_length ;
buf + = item_length ;
item_length = sprintf ( buf ,
" \n %d session %d share reconnects \n " ,
2007-06-06 00:35:06 +04:00
tcpSesReconnectCount . counter , tconInfoReconnectCount . counter ) ;
2005-04-17 02:20:36 +04:00
length + = item_length ;
buf + = item_length ;
item_length = sprintf ( buf ,
" Total vfs operations: %d maximum at one time: %d \n " ,
2007-06-06 00:35:06 +04:00
GlobalCurrentXid , GlobalMaxActiveXid ) ;
2005-04-17 02:20:36 +04:00
length + = item_length ;
buf + = item_length ;
i = 0 ;
read_lock ( & GlobalSMBSeslock ) ;
list_for_each ( tmp , & GlobalTreeConnectionList ) {
i + + ;
tcon = list_entry ( tmp , struct cifsTconInfo , cifsConnectionList ) ;
2007-06-06 00:35:06 +04:00
item_length = sprintf ( buf , " \n %d) %s " , i , tcon - > treeName ) ;
2005-04-17 02:20:36 +04:00
buf + = item_length ;
length + = item_length ;
2007-06-06 00:35:06 +04:00
if ( tcon - > tidStatus = = CifsNeedReconnect ) {
2005-04-17 02:20:36 +04:00
buf + = sprintf ( buf , " \t DISCONNECTED " ) ;
length + = 14 ;
}
2005-08-21 08:42:53 +04:00
item_length = sprintf ( buf , " \n SMBs: %d Oplock Breaks: %d " ,
2005-04-17 02:20:36 +04:00
atomic_read ( & tcon - > num_smbs_sent ) ,
atomic_read ( & tcon - > num_oplock_brks ) ) ;
buf + = item_length ;
length + = item_length ;
2005-10-10 21:57:19 +04:00
item_length = sprintf ( buf , " \n Reads: %d Bytes: %lld " ,
2005-04-17 02:20:36 +04:00
atomic_read ( & tcon - > num_reads ) ,
( long long ) ( tcon - > bytes_read ) ) ;
buf + = item_length ;
length + = item_length ;
2005-08-21 08:42:53 +04:00
item_length = sprintf ( buf , " \n Writes: %d Bytes: %lld " ,
2005-04-17 02:20:36 +04:00
atomic_read ( & tcon - > num_writes ) ,
( long long ) ( tcon - > bytes_written ) ) ;
2007-06-06 00:35:06 +04:00
buf + = item_length ;
length + = item_length ;
item_length = sprintf ( buf ,
2005-08-21 08:42:53 +04:00
" \n Locks: %d HardLinks: %d Symlinks: %d " ,
2007-06-06 00:35:06 +04:00
atomic_read ( & tcon - > num_locks ) ,
2005-08-21 08:42:53 +04:00
atomic_read ( & tcon - > num_hardlinks ) ,
atomic_read ( & tcon - > num_symlinks ) ) ;
2007-06-06 00:35:06 +04:00
buf + = item_length ;
length + = item_length ;
2005-08-21 08:42:53 +04:00
item_length = sprintf ( buf , " \n Opens: %d Closes: %d Deletes: %d " ,
atomic_read ( & tcon - > num_opens ) ,
atomic_read ( & tcon - > num_closes ) ,
atomic_read ( & tcon - > num_deletes ) ) ;
2005-04-17 02:20:36 +04:00
buf + = item_length ;
length + = item_length ;
2005-08-21 08:42:53 +04:00
item_length = sprintf ( buf , " \n Mkdirs: %d Rmdirs: %d " ,
2005-04-17 02:20:36 +04:00
atomic_read ( & tcon - > num_mkdirs ) ,
atomic_read ( & tcon - > num_rmdirs ) ) ;
buf + = item_length ;
length + = item_length ;
2005-08-21 08:42:53 +04:00
item_length = sprintf ( buf , " \n Renames: %d T2 Renames %d " ,
2005-04-17 02:20:36 +04:00
atomic_read ( & tcon - > num_renames ) ,
atomic_read ( & tcon - > num_t2renames ) ) ;
buf + = item_length ;
length + = item_length ;
2005-08-21 08:42:53 +04:00
item_length = sprintf ( buf , " \n FindFirst: %d FNext %d FClose %d " ,
2005-06-24 04:31:17 +04:00
atomic_read ( & tcon - > num_ffirst ) ,
atomic_read ( & tcon - > num_fnext ) ,
atomic_read ( & tcon - > num_fclose ) ) ;
buf + = item_length ;
length + = item_length ;
2005-04-17 02:20:36 +04:00
}
read_unlock ( & GlobalSMBSeslock ) ;
2007-06-06 00:35:06 +04:00
buf + = sprintf ( buf , " \n " ) ;
2005-04-17 02:20:36 +04:00
length + + ;
2007-06-06 00:35:06 +04:00
if ( offset + count > = length )
2005-04-17 02:20:36 +04:00
* eof = 1 ;
2007-06-06 00:35:06 +04:00
if ( length < offset ) {
2005-04-17 02:20:36 +04:00
* eof = 1 ;
return 0 ;
} else {
length = length - offset ;
}
if ( length > count )
length = count ;
2007-06-06 00:35:06 +04:00
2005-04-17 02:20:36 +04:00
return length ;
}
2008-02-12 23:32:36 +03:00
# endif /* STATS */
2005-04-17 02:20:36 +04:00
static struct proc_dir_entry * proc_fs_cifs ;
read_proc_t cifs_txanchor_read ;
static read_proc_t cifsFYI_read ;
static write_proc_t cifsFYI_write ;
static read_proc_t oplockEnabled_read ;
static write_proc_t oplockEnabled_write ;
static read_proc_t lookupFlag_read ;
static write_proc_t lookupFlag_write ;
static read_proc_t traceSMB_read ;
static write_proc_t traceSMB_write ;
static read_proc_t multiuser_mount_read ;
static write_proc_t multiuser_mount_write ;
2006-06-03 02:57:13 +04:00
static read_proc_t security_flags_read ;
static write_proc_t security_flags_write ;
2006-06-01 02:40:51 +04:00
/* static read_proc_t ntlmv2_enabled_read;
2005-04-17 02:20:36 +04:00
static write_proc_t ntlmv2_enabled_write ;
static read_proc_t packet_signing_enabled_read ;
2006-06-01 02:40:51 +04:00
static write_proc_t packet_signing_enabled_write ; */
2005-12-13 07:53:18 +03:00
static read_proc_t experimEnabled_read ;
static write_proc_t experimEnabled_write ;
2005-04-17 02:20:36 +04:00
static read_proc_t linuxExtensionsEnabled_read ;
static write_proc_t linuxExtensionsEnabled_write ;
void
cifs_proc_init ( void )
{
struct proc_dir_entry * pde ;
proc_fs_cifs = proc_mkdir ( " cifs " , proc_root_fs ) ;
if ( proc_fs_cifs = = NULL )
return ;
proc_fs_cifs - > owner = THIS_MODULE ;
create_proc_read_entry ( " DebugData " , 0 , proc_fs_cifs ,
cifs_debug_data_read , NULL ) ;
# ifdef CONFIG_CIFS_STATS
2005-10-12 06:58:06 +04:00
pde = create_proc_read_entry ( " Stats " , 0 , proc_fs_cifs ,
2005-04-17 02:20:36 +04:00
cifs_stats_read , NULL ) ;
2005-10-12 06:58:06 +04:00
if ( pde )
pde - > write_proc = cifs_stats_write ;
2008-02-12 23:32:36 +03:00
# endif /* STATS */
2005-04-17 02:20:36 +04:00
pde = create_proc_read_entry ( " cifsFYI " , 0 , proc_fs_cifs ,
cifsFYI_read , NULL ) ;
if ( pde )
pde - > write_proc = cifsFYI_write ;
pde =
create_proc_read_entry ( " traceSMB " , 0 , proc_fs_cifs ,
traceSMB_read , NULL ) ;
if ( pde )
pde - > write_proc = traceSMB_write ;
pde = create_proc_read_entry ( " OplockEnabled " , 0 , proc_fs_cifs ,
oplockEnabled_read , NULL ) ;
if ( pde )
pde - > write_proc = oplockEnabled_write ;
2005-06-24 04:31:17 +04:00
pde = create_proc_read_entry ( " Experimental " , 0 , proc_fs_cifs ,
2005-12-13 07:53:18 +03:00
experimEnabled_read , NULL ) ;
2005-04-17 02:20:36 +04:00
if ( pde )
2005-12-13 07:53:18 +03:00
pde - > write_proc = experimEnabled_write ;
2005-04-17 02:20:36 +04:00
pde = create_proc_read_entry ( " LinuxExtensionsEnabled " , 0 , proc_fs_cifs ,
linuxExtensionsEnabled_read , NULL ) ;
if ( pde )
pde - > write_proc = linuxExtensionsEnabled_write ;
pde =
create_proc_read_entry ( " MultiuserMount " , 0 , proc_fs_cifs ,
multiuser_mount_read , NULL ) ;
if ( pde )
pde - > write_proc = multiuser_mount_write ;
pde =
2006-06-01 23:20:10 +04:00
create_proc_read_entry ( " SecurityFlags " , 0 , proc_fs_cifs ,
2006-06-03 02:57:13 +04:00
security_flags_read , NULL ) ;
2005-04-17 02:20:36 +04:00
if ( pde )
2006-06-03 02:57:13 +04:00
pde - > write_proc = security_flags_write ;
2005-04-17 02:20:36 +04:00
pde =
create_proc_read_entry ( " LookupCacheEnabled " , 0 , proc_fs_cifs ,
lookupFlag_read , NULL ) ;
if ( pde )
pde - > write_proc = lookupFlag_write ;
2006-06-01 02:40:51 +04:00
/* pde =
2005-04-17 02:20:36 +04:00
create_proc_read_entry ( " NTLMV2Enabled " , 0 , proc_fs_cifs ,
ntlmv2_enabled_read , NULL ) ;
if ( pde )
pde - > write_proc = ntlmv2_enabled_write ;
pde =
create_proc_read_entry ( " PacketSigningEnabled " , 0 , proc_fs_cifs ,
packet_signing_enabled_read , NULL ) ;
if ( pde )
2006-06-01 02:40:51 +04:00
pde - > write_proc = packet_signing_enabled_write ; */
2005-04-17 02:20:36 +04:00
}
void
cifs_proc_clean ( void )
{
if ( proc_fs_cifs = = NULL )
return ;
remove_proc_entry ( " DebugData " , proc_fs_cifs ) ;
remove_proc_entry ( " cifsFYI " , proc_fs_cifs ) ;
remove_proc_entry ( " traceSMB " , proc_fs_cifs ) ;
# ifdef CONFIG_CIFS_STATS
remove_proc_entry ( " Stats " , proc_fs_cifs ) ;
# endif
remove_proc_entry ( " MultiuserMount " , proc_fs_cifs ) ;
remove_proc_entry ( " OplockEnabled " , proc_fs_cifs ) ;
2006-06-01 02:40:51 +04:00
/* remove_proc_entry("NTLMV2Enabled",proc_fs_cifs); */
2007-06-06 00:35:06 +04:00
remove_proc_entry ( " SecurityFlags " , proc_fs_cifs ) ;
/* remove_proc_entry("PacketSigningEnabled", proc_fs_cifs); */
remove_proc_entry ( " LinuxExtensionsEnabled " , proc_fs_cifs ) ;
remove_proc_entry ( " Experimental " , proc_fs_cifs ) ;
remove_proc_entry ( " LookupCacheEnabled " , proc_fs_cifs ) ;
2005-04-17 02:20:36 +04:00
remove_proc_entry ( " cifs " , proc_root_fs ) ;
}
static int
cifsFYI_read ( char * page , char * * start , off_t off , int count ,
int * eof , void * data )
{
int len ;
len = sprintf ( page , " %d \n " , cifsFYI ) ;
len - = off ;
* start = page + off ;
if ( len > count )
len = count ;
else
* eof = 1 ;
if ( len < 0 )
len = 0 ;
return len ;
}
static int
cifsFYI_write ( struct file * file , const char __user * buffer ,
unsigned long count , void * data )
{
char c ;
int rc ;
rc = get_user ( c , buffer ) ;
if ( rc )
return rc ;
if ( c = = ' 0 ' | | c = = ' n ' | | c = = ' N ' )
cifsFYI = 0 ;
else if ( c = = ' 1 ' | | c = = ' y ' | | c = = ' Y ' )
cifsFYI = 1 ;
2007-06-06 00:35:06 +04:00
else if ( ( c > ' 1 ' ) & & ( c < = ' 9 ' ) )
2005-10-12 06:58:06 +04:00
cifsFYI = ( int ) ( c - ' 0 ' ) ; /* see cifs_debug.h for meanings */
2005-04-17 02:20:36 +04:00
return count ;
}
static int
oplockEnabled_read ( char * page , char * * start , off_t off ,
int count , int * eof , void * data )
{
int len ;
len = sprintf ( page , " %d \n " , oplockEnabled ) ;
len - = off ;
* start = page + off ;
if ( len > count )
len = count ;
else
* eof = 1 ;
if ( len < 0 )
len = 0 ;
return len ;
}
static int
oplockEnabled_write ( struct file * file , const char __user * buffer ,
unsigned long count , void * data )
{
char c ;
int rc ;
rc = get_user ( c , buffer ) ;
if ( rc )
return rc ;
if ( c = = ' 0 ' | | c = = ' n ' | | c = = ' N ' )
oplockEnabled = 0 ;
else if ( c = = ' 1 ' | | c = = ' y ' | | c = = ' Y ' )
oplockEnabled = 1 ;
return count ;
}
static int
2005-12-13 07:53:18 +03:00
experimEnabled_read ( char * page , char * * start , off_t off ,
2007-06-06 00:35:06 +04:00
int count , int * eof , void * data )
2005-04-17 02:20:36 +04:00
{
2007-06-06 00:35:06 +04:00
int len ;
2005-04-17 02:20:36 +04:00
2007-06-06 00:35:06 +04:00
len = sprintf ( page , " %d \n " , experimEnabled ) ;
2005-12-13 07:53:18 +03:00
2007-06-06 00:35:06 +04:00
len - = off ;
* start = page + off ;
2005-04-17 02:20:36 +04:00
2007-06-06 00:35:06 +04:00
if ( len > count )
len = count ;
else
* eof = 1 ;
2005-04-17 02:20:36 +04:00
2007-06-06 00:35:06 +04:00
if ( len < 0 )
len = 0 ;
2005-04-17 02:20:36 +04:00
2007-06-06 00:35:06 +04:00
return len ;
2005-04-17 02:20:36 +04:00
}
static int
2005-12-13 07:53:18 +03:00
experimEnabled_write ( struct file * file , const char __user * buffer ,
2007-06-06 00:35:06 +04:00
unsigned long count , void * data )
2005-04-17 02:20:36 +04:00
{
2005-12-13 07:53:18 +03:00
char c ;
int rc ;
2005-04-17 02:20:36 +04:00
2005-12-13 07:53:18 +03:00
rc = get_user ( c , buffer ) ;
if ( rc )
return rc ;
if ( c = = ' 0 ' | | c = = ' n ' | | c = = ' N ' )
experimEnabled = 0 ;
else if ( c = = ' 1 ' | | c = = ' y ' | | c = = ' Y ' )
experimEnabled = 1 ;
else if ( c = = ' 2 ' )
experimEnabled = 2 ;
2005-04-17 02:20:36 +04:00
2005-12-13 07:53:18 +03:00
return count ;
2005-04-17 02:20:36 +04:00
}
static int
linuxExtensionsEnabled_read ( char * page , char * * start , off_t off ,
2007-06-06 00:35:06 +04:00
int count , int * eof , void * data )
2005-04-17 02:20:36 +04:00
{
2007-06-06 00:35:06 +04:00
int len ;
2005-04-17 02:20:36 +04:00
2007-06-06 00:35:06 +04:00
len = sprintf ( page , " %d \n " , linuxExtEnabled ) ;
len - = off ;
* start = page + off ;
2005-04-17 02:20:36 +04:00
2007-06-06 00:35:06 +04:00
if ( len > count )
len = count ;
else
* eof = 1 ;
2005-04-17 02:20:36 +04:00
2007-06-06 00:35:06 +04:00
if ( len < 0 )
len = 0 ;
2005-04-17 02:20:36 +04:00
2007-06-06 00:35:06 +04:00
return len ;
2005-04-17 02:20:36 +04:00
}
static int
linuxExtensionsEnabled_write ( struct file * file , const char __user * buffer ,
2007-06-06 00:35:06 +04:00
unsigned long count , void * data )
2005-04-17 02:20:36 +04:00
{
2007-06-06 00:35:06 +04:00
char c ;
int rc ;
rc = get_user ( c , buffer ) ;
if ( rc )
return rc ;
if ( c = = ' 0 ' | | c = = ' n ' | | c = = ' N ' )
linuxExtEnabled = 0 ;
else if ( c = = ' 1 ' | | c = = ' y ' | | c = = ' Y ' )
linuxExtEnabled = 1 ;
return count ;
2005-04-17 02:20:36 +04:00
}
static int
lookupFlag_read ( char * page , char * * start , off_t off ,
2007-06-06 00:35:06 +04:00
int count , int * eof , void * data )
2005-04-17 02:20:36 +04:00
{
int len ;
len = sprintf ( page , " %d \n " , lookupCacheEnabled ) ;
len - = off ;
* start = page + off ;
if ( len > count )
len = count ;
else
* eof = 1 ;
if ( len < 0 )
len = 0 ;
return len ;
}
static int
lookupFlag_write ( struct file * file , const char __user * buffer ,
unsigned long count , void * data )
{
char c ;
int rc ;
rc = get_user ( c , buffer ) ;
if ( rc )
return rc ;
if ( c = = ' 0 ' | | c = = ' n ' | | c = = ' N ' )
lookupCacheEnabled = 0 ;
else if ( c = = ' 1 ' | | c = = ' y ' | | c = = ' Y ' )
lookupCacheEnabled = 1 ;
return count ;
}
static int
traceSMB_read ( char * page , char * * start , off_t off , int count ,
int * eof , void * data )
{
int len ;
len = sprintf ( page , " %d \n " , traceSMB ) ;
len - = off ;
* start = page + off ;
if ( len > count )
len = count ;
else
* eof = 1 ;
if ( len < 0 )
len = 0 ;
return len ;
}
static int
traceSMB_write ( struct file * file , const char __user * buffer ,
unsigned long count , void * data )
{
char c ;
int rc ;
rc = get_user ( c , buffer ) ;
if ( rc )
return rc ;
if ( c = = ' 0 ' | | c = = ' n ' | | c = = ' N ' )
traceSMB = 0 ;
else if ( c = = ' 1 ' | | c = = ' y ' | | c = = ' Y ' )
traceSMB = 1 ;
return count ;
}
static int
multiuser_mount_read ( char * page , char * * start , off_t off ,
int count , int * eof , void * data )
{
int len ;
len = sprintf ( page , " %d \n " , multiuser_mount ) ;
len - = off ;
* start = page + off ;
if ( len > count )
len = count ;
else
* eof = 1 ;
if ( len < 0 )
len = 0 ;
return len ;
}
static int
multiuser_mount_write ( struct file * file , const char __user * buffer ,
unsigned long count , void * data )
{
char c ;
int rc ;
rc = get_user ( c , buffer ) ;
if ( rc )
return rc ;
if ( c = = ' 0 ' | | c = = ' n ' | | c = = ' N ' )
multiuser_mount = 0 ;
else if ( c = = ' 1 ' | | c = = ' y ' | | c = = ' Y ' )
multiuser_mount = 1 ;
return count ;
}
static int
2006-06-03 02:57:13 +04:00
security_flags_read ( char * page , char * * start , off_t off ,
2005-04-17 02:20:36 +04:00
int count , int * eof , void * data )
{
int len ;
2006-06-01 02:40:51 +04:00
len = sprintf ( page , " 0x%x \n " , extended_security ) ;
2005-04-17 02:20:36 +04:00
len - = off ;
* start = page + off ;
if ( len > count )
len = count ;
else
* eof = 1 ;
if ( len < 0 )
len = 0 ;
return len ;
}
static int
2006-06-03 02:57:13 +04:00
security_flags_write ( struct file * file , const char __user * buffer ,
2005-04-17 02:20:36 +04:00
unsigned long count , void * data )
{
2006-06-03 02:57:13 +04:00
unsigned int flags ;
char flags_string [ 12 ] ;
2005-04-17 02:20:36 +04:00
char c ;
2006-06-03 02:57:13 +04:00
2007-06-06 00:35:06 +04:00
if ( ( count < 1 ) | | ( count > 11 ) )
2006-06-01 02:40:51 +04:00
return - EINVAL ;
2005-04-17 02:20:36 +04:00
2006-06-03 02:57:13 +04:00
memset ( flags_string , 0 , 12 ) ;
2006-06-01 02:40:51 +04:00
2007-06-06 00:35:06 +04:00
if ( copy_from_user ( flags_string , buffer , count ) )
2006-06-03 02:57:13 +04:00
return - EFAULT ;
2006-06-01 02:40:51 +04:00
2007-06-06 00:35:06 +04:00
if ( count < 3 ) {
2006-06-03 02:57:13 +04:00
/* single char or single char followed by null */
c = flags_string [ 0 ] ;
2007-10-05 00:05:09 +04:00
if ( c = = ' 0 ' | | c = = ' n ' | | c = = ' N ' ) {
2006-06-03 02:57:13 +04:00
extended_security = CIFSSEC_DEF ; /* default */
2007-10-05 00:05:09 +04:00
return count ;
} else if ( c = = ' 1 ' | | c = = ' y ' | | c = = ' Y ' ) {
2006-06-03 02:57:13 +04:00
extended_security = CIFSSEC_MAX ;
2007-10-05 00:05:09 +04:00
return count ;
} else if ( ! isdigit ( c ) ) {
cERROR ( 1 , ( " invalid flag %c " , c ) ) ;
return - EINVAL ;
}
2006-06-03 02:57:13 +04:00
}
/* else we have a number */
flags = simple_strtoul ( flags_string , NULL , 0 ) ;
2007-06-06 00:35:06 +04:00
cFYI ( 1 , ( " sec flags 0x%x " , flags ) ) ;
2006-06-03 02:57:13 +04:00
2007-06-06 00:35:06 +04:00
if ( flags < = 0 ) {
cERROR ( 1 , ( " invalid security flags %s " , flags_string ) ) ;
2006-06-03 02:57:13 +04:00
return - EINVAL ;
}
2005-04-17 02:20:36 +04:00
2007-06-06 00:35:06 +04:00
if ( flags & ~ CIFSSEC_MASK ) {
cERROR ( 1 , ( " attempt to set unsupported security flags 0x%x " ,
2006-06-03 02:57:13 +04:00
flags & ~ CIFSSEC_MASK ) ) ;
return - EINVAL ;
}
/* flags look ok - update the global security flags for cifs module */
extended_security = flags ;
2007-06-28 22:41:42 +04:00
if ( extended_security & CIFSSEC_MUST_SIGN ) {
/* requiring signing implies signing is allowed */
extended_security | = CIFSSEC_MAY_SIGN ;
cFYI ( 1 , ( " packet signing now required " ) ) ;
} else if ( ( extended_security & CIFSSEC_MAY_SIGN ) = = 0 ) {
cFYI ( 1 , ( " packet signing disabled " ) ) ;
}
/* BB should we turn on MAY flags for other MUST options? */
2005-04-17 02:20:36 +04:00
return count ;
}
2008-02-12 23:32:36 +03:00
# else
static inline void cifs_proc_init ( void )
{
}
static inline void cifs_proc_clean ( void )
{
}
# endif /* PROC_FS */