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 ) ;
2008-08-09 01:10:16 +04:00
cERROR ( 1 , ( " State: %d Cmd: %d Pid: %d Tsk: %p Mid %d " ,
mid_entry - > midState ,
( int ) mid_entry - > command ,
mid_entry - > pid ,
mid_entry - > tsk ,
mid_entry - > mid ) ) ;
2006-06-01 02:40:51 +04:00
# ifdef CONFIG_CIFS_STATS2
2008-08-09 01:10:16 +04:00
cERROR ( 1 , ( " IsLarge: %d buf: %p time rcv: %ld now: %ld " ,
mid_entry - > largeBuf ,
mid_entry - > resp_buf ,
mid_entry - > when_received ,
jiffies ) ) ;
2006-06-01 02:40:51 +04:00
# endif /* STATS2 */
2008-08-09 01:10:16 +04:00
cERROR ( 1 , ( " IsMult: %d IsEnd: %d " , mid_entry - > multiRsp ,
mid_entry - > multiEnd ) ) ;
if ( mid_entry - > resp_buf ) {
cifs_dump_detail ( mid_entry - > resp_buf ) ;
cifs_dump_mem ( " existing buf: " ,
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
2008-07-24 06:34:24 +04:00
static int cifs_debug_data_proc_show ( struct seq_file * m , void * v )
2005-04-17 02:20:36 +04:00
{
2008-11-15 19:12:47 +03:00
struct list_head * tmp1 , * tmp2 , * tmp3 ;
2007-06-25 01:15:44 +04:00
struct mid_q_entry * mid_entry ;
2008-11-14 21:53:46 +03:00
struct TCP_Server_Info * server ;
2005-04-17 02:20:36 +04:00
struct cifsSesInfo * ses ;
struct cifsTconInfo * tcon ;
2008-11-15 19:12:47 +03:00
int i , j ;
__u32 dev_type ;
2005-04-17 02:20:36 +04:00
2008-07-24 06:34:24 +04:00
seq_puts ( m ,
2005-04-17 02:20:36 +04:00
" Display Internal CIFS Data Structures for Debugging \n "
" --------------------------------------------------- \n " ) ;
2008-07-24 06:34:24 +04:00
seq_printf ( m , " CIFS Version %s \n " , CIFS_VERSION ) ;
seq_printf ( m , " Active VFS Requests: %d \n " , GlobalTotalActiveXid ) ;
seq_printf ( m , " Servers: " ) ;
2005-04-17 02:20:36 +04:00
i = 0 ;
2008-11-14 21:53:46 +03:00
read_lock ( & cifs_tcp_ses_lock ) ;
2008-11-15 19:12:47 +03:00
list_for_each ( tmp1 , & cifs_tcp_ses_list ) {
server = list_entry ( tmp1 , struct TCP_Server_Info ,
2008-11-14 21:53:46 +03:00
tcp_ses_list ) ;
2005-04-17 02:20:36 +04:00
i + + ;
2008-11-14 21:53:46 +03:00
list_for_each ( tmp2 , & server - > smb_ses_list ) {
ses = list_entry ( tmp2 , struct cifsSesInfo ,
smb_ses_list ) ;
if ( ( ses - > serverDomain = = NULL ) | |
( ses - > serverOS = = NULL ) | |
( ses - > serverNOS = = NULL ) ) {
2008-11-15 19:12:47 +03:00
seq_printf ( m , " \n %d) entry for %s not fully "
" displayed \n \t " , i , ses - > serverName ) ;
2008-11-14 21:53:46 +03:00
} else {
seq_printf ( m ,
2008-11-15 19:12:47 +03:00
" \n %d) Name: %s Domain: %s Uses: %d OS: "
" %s \n \t NOS: %s \t Capability: 0x%x \n \t SMB "
2007-06-06 00:35:06 +04:00
" session status: %d \t " ,
2005-04-29 09:41:07 +04:00
i , ses - > serverName , ses - > serverDomain ,
2008-11-14 21:53:46 +03:00
ses - > ses_count , ses - > serverOS , ses - > serverNOS ,
2007-06-06 00:35:06 +04:00
ses - > capabilities , ses - > status ) ;
2008-11-14 21:53:46 +03:00
}
2008-07-24 06:34:24 +04:00
seq_printf ( m , " TCP status: %d \n \t Local Users To "
2008-11-14 21:53:46 +03:00
" Server: %d SecMode: 0x%x Req On Wire: %d " ,
server - > tcpStatus , server - > srv_count ,
server - > secMode ,
atomic_read ( & server - > inFlight ) ) ;
2005-10-07 20:51:05 +04:00
# ifdef CONFIG_CIFS_STATS2
2008-07-24 06:34:24 +04:00
seq_printf ( m , " In Send: %d In MaxReq Wait: %d " ,
2008-11-14 21:53:46 +03:00
atomic_read ( & server - > inSend ) ,
atomic_read ( & server - > num_waiters ) ) ;
2005-10-07 20:51:05 +04:00
# endif
2008-11-15 19:12:47 +03:00
seq_puts ( m , " \n \t Shares: " ) ;
j = 0 ;
list_for_each ( tmp3 , & ses - > tcon_list ) {
tcon = list_entry ( tmp3 , struct cifsTconInfo ,
tcon_list ) ;
+ + j ;
dev_type = le32_to_cpu ( tcon - > fsDevInfo . DeviceType ) ;
seq_printf ( m , " \n \t %d) %s Mounts: %d " , j ,
tcon - > treeName , tcon - > tc_count ) ;
if ( tcon - > nativeFileSystem ) {
seq_printf ( m , " Type: %s " ,
tcon - > nativeFileSystem ) ;
}
seq_printf ( m , " DevInfo: 0x%x Attributes: 0x%x "
" \n PathComponentMax: %d Status: 0x%d " ,
le32_to_cpu ( tcon - > fsDevInfo . DeviceCharacteristics ) ,
le32_to_cpu ( tcon - > fsAttrInfo . Attributes ) ,
le32_to_cpu ( tcon - > fsAttrInfo . MaxPathNameComponentLength ) ,
tcon - > tidStatus ) ;
if ( dev_type = = FILE_DEVICE_DISK )
seq_puts ( m , " type: DISK " ) ;
else if ( dev_type = = FILE_DEVICE_CD_ROM )
seq_puts ( m , " type: CDROM " ) ;
else
seq_printf ( m , " type: %d " , dev_type ) ;
if ( tcon - > need_reconnect )
seq_puts ( m , " \t DISCONNECTED " ) ;
seq_putc ( m , ' \n ' ) ;
}
seq_puts ( m , " \n \t MIDs: \n " ) ;
2005-04-17 02:20:36 +04:00
spin_lock ( & GlobalMid_Lock ) ;
2008-11-14 21:53:46 +03:00
list_for_each ( tmp3 , & server - > pending_mid_q ) {
2008-11-15 19:12:47 +03:00
mid_entry = list_entry ( tmp3 , struct mid_q_entry ,
2005-04-17 02:20:36 +04:00
qhead ) ;
2008-11-15 19:12:47 +03:00
seq_printf ( m , " \t State: %d com: %d pid: "
2008-08-09 01:10:16 +04:00
" %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
}
2007-06-06 00:35:06 +04:00
spin_unlock ( & GlobalMid_Lock ) ;
2005-04-17 02:20:36 +04:00
}
}
2008-11-14 21:53:46 +03:00
read_unlock ( & cifs_tcp_ses_lock ) ;
2008-07-24 06:34:24 +04:00
seq_putc ( m , ' \n ' ) ;
2005-04-17 02:20:36 +04:00
/* BB add code to dump additional info such as TCP session info now */
2008-07-24 06:34:24 +04:00
return 0 ;
}
2005-04-17 02:20:36 +04:00
2008-07-24 06:34:24 +04:00
static int cifs_debug_data_proc_open ( struct inode * inode , struct file * file )
{
return single_open ( file , cifs_debug_data_proc_show , NULL ) ;
2005-04-17 02:20:36 +04:00
}
2008-07-24 06:34:24 +04:00
static const struct file_operations cifs_debug_data_proc_fops = {
. owner = THIS_MODULE ,
. open = cifs_debug_data_proc_open ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
} ;
2005-10-12 06:58:06 +04:00
2008-07-24 06:34:24 +04:00
# ifdef CONFIG_CIFS_STATS
static ssize_t cifs_stats_proc_write ( struct file * file ,
const char __user * buffer , size_t count , loff_t * ppos )
2005-10-12 06:58:06 +04:00
{
2007-06-06 00:35:06 +04:00
char c ;
int rc ;
2008-11-15 19:12:47 +03:00
struct list_head * tmp1 , * tmp2 , * tmp3 ;
struct TCP_Server_Info * server ;
struct cifsSesInfo * ses ;
2005-10-12 06:58:06 +04:00
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-12-04 00:58:57 +03:00
# ifdef CONFIG_CIFS_STATS2
atomic_set ( & totBufAllocCount , 0 ) ;
atomic_set ( & totSmBufAllocCount , 0 ) ;
# endif /* CONFIG_CIFS_STATS2 */
2008-11-15 19:12:47 +03:00
read_lock ( & cifs_tcp_ses_lock ) ;
list_for_each ( tmp1 , & cifs_tcp_ses_list ) {
server = list_entry ( tmp1 , struct TCP_Server_Info ,
tcp_ses_list ) ;
2008-11-17 06:57:13 +03:00
list_for_each ( tmp2 , & server - > smb_ses_list ) {
2008-11-15 19:12:47 +03:00
ses = list_entry ( tmp2 , struct cifsSesInfo ,
2008-11-17 06:57:13 +03:00
smb_ses_list ) ;
2008-11-15 19:12:47 +03:00
list_for_each ( tmp3 , & ses - > tcon_list ) {
tcon = list_entry ( tmp3 ,
struct cifsTconInfo ,
tcon_list ) ;
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 ) ;
2009-07-10 19:27:25 +04:00
atomic_set ( & tcon - > num_posixopens , 0 ) ;
atomic_set ( & tcon - > num_posixmkdirs , 0 ) ;
2008-11-15 19:12:47 +03:00
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 ) ;
}
}
2005-10-12 06:58:06 +04:00
}
2008-11-15 19:12:47 +03:00
read_unlock ( & cifs_tcp_ses_lock ) ;
2005-10-12 06:58:06 +04:00
}
2007-06-06 00:35:06 +04:00
return count ;
2005-10-12 06:58:06 +04:00
}
2008-07-24 06:34:24 +04:00
static int cifs_stats_proc_show ( struct seq_file * m , void * v )
2005-04-17 02:20:36 +04:00
{
2008-07-24 06:34:24 +04:00
int i ;
2008-11-15 19:12:47 +03:00
struct list_head * tmp1 , * tmp2 , * tmp3 ;
struct TCP_Server_Info * server ;
struct cifsSesInfo * ses ;
2005-04-17 02:20:36 +04:00
struct cifsTconInfo * tcon ;
2008-07-24 06:34:24 +04:00
seq_printf ( m ,
2005-04-17 02:20:36 +04:00
" Resources in use \n CIFS Session: %d \n " ,
sesInfoAllocCount . counter ) ;
2008-07-24 06:34:24 +04:00
seq_printf ( m , " Share (unique mount targets): %d \n " ,
2005-04-17 02:20:36 +04:00
tconInfoAllocCount . counter ) ;
2008-07-24 06:34:24 +04:00
seq_printf ( m , " SMB Request/Response Buffer: %d Pool size: %d \n " ,
2005-04-29 09:41:07 +04:00
bufAllocCount . counter ,
cifs_min_rcv + tcpSesAllocCount . counter ) ;
2008-07-24 06:34:24 +04:00
seq_printf ( m , " SMB Small Req/Resp Buffer: %d Pool size: %d \n " ,
2007-06-06 00:35:06 +04:00
smBufAllocCount . counter , cifs_min_small ) ;
2005-12-04 01:11:37 +03:00
# ifdef CONFIG_CIFS_STATS2
2008-07-24 06:34:24 +04:00
seq_printf ( m , " 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
# endif /* CONFIG_CIFS_STATS2 */
2008-07-24 06:34:24 +04:00
seq_printf ( m , " Operations (MIDs): %d \n " , midCount . counter ) ;
seq_printf ( m ,
2005-04-17 02:20:36 +04:00
" \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
2008-07-24 06:34:24 +04:00
seq_printf ( m ,
2005-04-17 02:20:36 +04:00
" 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
i = 0 ;
2008-11-15 19:12:47 +03:00
read_lock ( & cifs_tcp_ses_lock ) ;
list_for_each ( tmp1 , & cifs_tcp_ses_list ) {
server = list_entry ( tmp1 , struct TCP_Server_Info ,
tcp_ses_list ) ;
list_for_each ( tmp2 , & server - > smb_ses_list ) {
ses = list_entry ( tmp2 , struct cifsSesInfo ,
smb_ses_list ) ;
list_for_each ( tmp3 , & ses - > tcon_list ) {
tcon = list_entry ( tmp3 ,
struct cifsTconInfo ,
tcon_list ) ;
i + + ;
seq_printf ( m , " \n %d) %s " , i , tcon - > treeName ) ;
if ( tcon - > need_reconnect )
seq_puts ( m , " \t DISCONNECTED " ) ;
seq_printf ( m , " \n SMBs: %d Oplock Breaks: %d " ,
atomic_read ( & tcon - > num_smbs_sent ) ,
atomic_read ( & tcon - > num_oplock_brks ) ) ;
seq_printf ( m , " \n Reads: %d Bytes: %lld " ,
atomic_read ( & tcon - > num_reads ) ,
( long long ) ( tcon - > bytes_read ) ) ;
seq_printf ( m , " \n Writes: %d Bytes: %lld " ,
atomic_read ( & tcon - > num_writes ) ,
( long long ) ( tcon - > bytes_written ) ) ;
2009-02-22 00:17:43 +03:00
seq_printf ( m , " \n Flushes: %d " ,
atomic_read ( & tcon - > num_flushes ) ) ;
2008-11-15 19:12:47 +03:00
seq_printf ( m , " \n Locks: %d HardLinks: %d "
" Symlinks: %d " ,
atomic_read ( & tcon - > num_locks ) ,
atomic_read ( & tcon - > num_hardlinks ) ,
atomic_read ( & tcon - > num_symlinks ) ) ;
2009-07-10 19:27:25 +04:00
seq_printf ( m , " \n Opens: %d Closes: %d "
2008-11-15 19:12:47 +03:00
" Deletes: %d " ,
atomic_read ( & tcon - > num_opens ) ,
atomic_read ( & tcon - > num_closes ) ,
atomic_read ( & tcon - > num_deletes ) ) ;
2009-07-10 19:27:25 +04:00
seq_printf ( m , " \n Posix Opens: %d "
" Posix Mkdirs: %d " ,
atomic_read ( & tcon - > num_posixopens ) ,
atomic_read ( & tcon - > num_posixmkdirs ) ) ;
2008-11-15 19:12:47 +03:00
seq_printf ( m , " \n Mkdirs: %d Rmdirs: %d " ,
atomic_read ( & tcon - > num_mkdirs ) ,
atomic_read ( & tcon - > num_rmdirs ) ) ;
seq_printf ( m , " \n Renames: %d T2 Renames %d " ,
atomic_read ( & tcon - > num_renames ) ,
atomic_read ( & tcon - > num_t2renames ) ) ;
seq_printf ( m , " \n FindFirst: %d FNext %d "
" FClose %d " ,
atomic_read ( & tcon - > num_ffirst ) ,
atomic_read ( & tcon - > num_fnext ) ,
atomic_read ( & tcon - > num_fclose ) ) ;
}
}
2005-04-17 02:20:36 +04:00
}
2008-11-15 19:12:47 +03:00
read_unlock ( & cifs_tcp_ses_lock ) ;
2005-04-17 02:20:36 +04:00
2008-07-24 06:34:24 +04:00
seq_putc ( m , ' \n ' ) ;
return 0 ;
}
2007-06-06 00:35:06 +04:00
2008-07-24 06:34:24 +04:00
static int cifs_stats_proc_open ( struct inode * inode , struct file * file )
{
return single_open ( file , cifs_stats_proc_show , NULL ) ;
2005-04-17 02:20:36 +04:00
}
2008-07-24 06:34:24 +04:00
static const struct file_operations cifs_stats_proc_fops = {
. owner = THIS_MODULE ,
. open = cifs_stats_proc_open ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
. write = cifs_stats_proc_write ,
} ;
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 ;
2008-07-24 06:34:24 +04:00
static const struct file_operations cifsFYI_proc_fops ;
static const struct file_operations cifs_oplock_proc_fops ;
static const struct file_operations cifs_lookup_cache_proc_fops ;
static const struct file_operations traceSMB_proc_fops ;
static const struct file_operations cifs_multiuser_mount_proc_fops ;
static const struct file_operations cifs_security_flags_proc_fops ;
static const struct file_operations cifs_experimental_proc_fops ;
static const struct file_operations cifs_linux_ext_proc_fops ;
2005-04-17 02:20:36 +04:00
void
cifs_proc_init ( void )
{
2008-04-29 12:01:42 +04:00
proc_fs_cifs = proc_mkdir ( " fs/cifs " , NULL ) ;
2005-04-17 02:20:36 +04:00
if ( proc_fs_cifs = = NULL )
return ;
2008-07-24 06:34:24 +04:00
proc_create ( " DebugData " , 0 , proc_fs_cifs , & cifs_debug_data_proc_fops ) ;
2005-04-17 02:20:36 +04:00
# ifdef CONFIG_CIFS_STATS
2008-07-24 06:34:24 +04:00
proc_create ( " Stats " , 0 , proc_fs_cifs , & cifs_stats_proc_fops ) ;
2008-02-12 23:32:36 +03:00
# endif /* STATS */
2008-07-24 06:34:24 +04:00
proc_create ( " cifsFYI " , 0 , proc_fs_cifs , & cifsFYI_proc_fops ) ;
proc_create ( " traceSMB " , 0 , proc_fs_cifs , & traceSMB_proc_fops ) ;
proc_create ( " OplockEnabled " , 0 , proc_fs_cifs , & cifs_oplock_proc_fops ) ;
2008-07-24 06:37:45 +04:00
proc_create ( " Experimental " , 0 , proc_fs_cifs ,
& cifs_experimental_proc_fops ) ;
proc_create ( " LinuxExtensionsEnabled " , 0 , proc_fs_cifs ,
& cifs_linux_ext_proc_fops ) ;
proc_create ( " MultiuserMount " , 0 , proc_fs_cifs ,
& cifs_multiuser_mount_proc_fops ) ;
proc_create ( " SecurityFlags " , 0 , proc_fs_cifs ,
& cifs_security_flags_proc_fops ) ;
proc_create ( " LookupCacheEnabled " , 0 , proc_fs_cifs ,
& cifs_lookup_cache_proc_fops ) ;
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 ) ;
2007-06-06 00:35:06 +04:00
remove_proc_entry ( " SecurityFlags " , proc_fs_cifs ) ;
remove_proc_entry ( " LinuxExtensionsEnabled " , proc_fs_cifs ) ;
remove_proc_entry ( " Experimental " , proc_fs_cifs ) ;
remove_proc_entry ( " LookupCacheEnabled " , proc_fs_cifs ) ;
2008-04-29 12:01:42 +04:00
remove_proc_entry ( " fs/cifs " , NULL ) ;
2005-04-17 02:20:36 +04:00
}
2008-07-24 06:34:24 +04:00
static int cifsFYI_proc_show ( struct seq_file * m , void * v )
2005-04-17 02:20:36 +04:00
{
2008-07-24 06:34:24 +04:00
seq_printf ( m , " %d \n " , cifsFYI ) ;
return 0 ;
}
2005-04-17 02:20:36 +04:00
2008-07-24 06:34:24 +04:00
static int cifsFYI_proc_open ( struct inode * inode , struct file * file )
{
return single_open ( file , cifsFYI_proc_show , NULL ) ;
2005-04-17 02:20:36 +04:00
}
2008-07-24 06:34:24 +04:00
static ssize_t cifsFYI_proc_write ( struct file * file , const char __user * buffer ,
size_t count , loff_t * ppos )
2005-04-17 02:20:36 +04:00
{
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 ;
}
2008-07-24 06:34:24 +04:00
static const struct file_operations cifsFYI_proc_fops = {
. owner = THIS_MODULE ,
. open = cifsFYI_proc_open ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
. write = cifsFYI_proc_write ,
} ;
2005-04-17 02:20:36 +04:00
2008-07-24 06:34:24 +04:00
static int cifs_oplock_proc_show ( struct seq_file * m , void * v )
{
seq_printf ( m , " %d \n " , oplockEnabled ) ;
return 0 ;
}
2005-04-17 02:20:36 +04:00
2008-07-24 06:34:24 +04:00
static int cifs_oplock_proc_open ( struct inode * inode , struct file * file )
{
return single_open ( file , cifs_oplock_proc_show , NULL ) ;
2005-04-17 02:20:36 +04:00
}
2008-07-24 06:34:24 +04:00
static ssize_t cifs_oplock_proc_write ( struct file * file ,
const char __user * buffer , size_t count , loff_t * ppos )
2005-04-17 02:20:36 +04:00
{
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 ;
}
2008-07-24 06:34:24 +04:00
static const struct file_operations cifs_oplock_proc_fops = {
. owner = THIS_MODULE ,
. open = cifs_oplock_proc_open ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
. write = cifs_oplock_proc_write ,
} ;
2005-04-17 02:20:36 +04:00
2008-07-24 06:34:24 +04:00
static int cifs_experimental_proc_show ( struct seq_file * m , void * v )
{
seq_printf ( m , " %d \n " , experimEnabled ) ;
return 0 ;
}
2005-04-17 02:20:36 +04:00
2008-07-24 06:34:24 +04:00
static int cifs_experimental_proc_open ( struct inode * inode , struct file * file )
{
return single_open ( file , cifs_experimental_proc_show , NULL ) ;
2005-04-17 02:20:36 +04:00
}
2008-07-24 06:34:24 +04:00
static ssize_t cifs_experimental_proc_write ( struct file * file ,
const char __user * buffer , size_t count , loff_t * ppos )
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
}
2008-07-24 06:34:24 +04:00
static const struct file_operations cifs_experimental_proc_fops = {
. owner = THIS_MODULE ,
. open = cifs_experimental_proc_open ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
. write = cifs_experimental_proc_write ,
} ;
2005-04-17 02:20:36 +04:00
2008-07-24 06:34:24 +04:00
static int cifs_linux_ext_proc_show ( struct seq_file * m , void * v )
{
seq_printf ( m , " %d \n " , linuxExtEnabled ) ;
return 0 ;
}
2005-04-17 02:20:36 +04:00
2008-07-24 06:34:24 +04:00
static int cifs_linux_ext_proc_open ( struct inode * inode , struct file * file )
{
return single_open ( file , cifs_linux_ext_proc_show , NULL ) ;
2005-04-17 02:20:36 +04:00
}
2008-07-24 06:34:24 +04:00
static ssize_t cifs_linux_ext_proc_write ( struct file * file ,
const char __user * buffer , size_t count , loff_t * ppos )
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
}
2008-07-24 06:34:24 +04:00
static const struct file_operations cifs_linux_ext_proc_fops = {
. owner = THIS_MODULE ,
. open = cifs_linux_ext_proc_open ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
. write = cifs_linux_ext_proc_write ,
} ;
2005-04-17 02:20:36 +04:00
2008-07-24 06:34:24 +04:00
static int cifs_lookup_cache_proc_show ( struct seq_file * m , void * v )
2005-04-17 02:20:36 +04:00
{
2008-07-24 06:34:24 +04:00
seq_printf ( m , " %d \n " , lookupCacheEnabled ) ;
return 0 ;
}
2005-04-17 02:20:36 +04:00
2008-07-24 06:34:24 +04:00
static int cifs_lookup_cache_proc_open ( struct inode * inode , struct file * file )
{
return single_open ( file , cifs_lookup_cache_proc_show , NULL ) ;
2005-04-17 02:20:36 +04:00
}
2008-07-24 06:34:24 +04:00
static ssize_t cifs_lookup_cache_proc_write ( struct file * file ,
const char __user * buffer , size_t count , loff_t * ppos )
2005-04-17 02:20:36 +04:00
{
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 ;
}
2008-07-24 06:34:24 +04:00
static const struct file_operations cifs_lookup_cache_proc_fops = {
. owner = THIS_MODULE ,
. open = cifs_lookup_cache_proc_open ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
. write = cifs_lookup_cache_proc_write ,
} ;
2005-04-17 02:20:36 +04:00
2008-07-24 06:34:24 +04:00
static int traceSMB_proc_show ( struct seq_file * m , void * v )
{
seq_printf ( m , " %d \n " , traceSMB ) ;
return 0 ;
}
2005-04-17 02:20:36 +04:00
2008-07-24 06:34:24 +04:00
static int traceSMB_proc_open ( struct inode * inode , struct file * file )
{
return single_open ( file , traceSMB_proc_show , NULL ) ;
2005-04-17 02:20:36 +04:00
}
2008-07-24 06:34:24 +04:00
static ssize_t traceSMB_proc_write ( struct file * file , const char __user * buffer ,
size_t count , loff_t * ppos )
2005-04-17 02:20:36 +04:00
{
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 ;
}
2008-07-24 06:34:24 +04:00
static const struct file_operations traceSMB_proc_fops = {
. owner = THIS_MODULE ,
. open = traceSMB_proc_open ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
. write = traceSMB_proc_write ,
} ;
2005-04-17 02:20:36 +04:00
2008-07-24 06:34:24 +04:00
static int cifs_multiuser_mount_proc_show ( struct seq_file * m , void * v )
{
seq_printf ( m , " %d \n " , multiuser_mount ) ;
return 0 ;
}
2005-04-17 02:20:36 +04:00
2008-07-24 06:37:45 +04:00
static int cifs_multiuser_mount_proc_open ( struct inode * inode , struct file * fh )
2008-07-24 06:34:24 +04:00
{
2008-07-24 06:37:45 +04:00
return single_open ( fh , cifs_multiuser_mount_proc_show , NULL ) ;
2005-04-17 02:20:36 +04:00
}
2008-07-24 06:34:24 +04:00
static ssize_t cifs_multiuser_mount_proc_write ( struct file * file ,
const char __user * buffer , size_t count , loff_t * ppos )
2005-04-17 02:20:36 +04:00
{
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 ;
}
2008-07-24 06:34:24 +04:00
static const struct file_operations cifs_multiuser_mount_proc_fops = {
. owner = THIS_MODULE ,
. open = cifs_multiuser_mount_proc_open ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
. write = cifs_multiuser_mount_proc_write ,
} ;
2005-04-17 02:20:36 +04:00
2008-07-24 06:34:24 +04:00
static int cifs_security_flags_proc_show ( struct seq_file * m , void * v )
{
seq_printf ( m , " 0x%x \n " , extended_security ) ;
return 0 ;
}
2005-04-17 02:20:36 +04:00
2008-07-24 06:34:24 +04:00
static int cifs_security_flags_proc_open ( struct inode * inode , struct file * file )
{
return single_open ( file , cifs_security_flags_proc_show , NULL ) ;
2005-04-17 02:20:36 +04:00
}
2008-07-24 06:34:24 +04:00
static ssize_t cifs_security_flags_proc_write ( struct file * file ,
const char __user * buffer , size_t count , loff_t * ppos )
2005-04-17 02:20:36 +04:00
{
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-07-24 06:34:24 +04:00
static const struct file_operations cifs_security_flags_proc_fops = {
. owner = THIS_MODULE ,
. open = cifs_security_flags_proc_open ,
. read = seq_read ,
. llseek = seq_lseek ,
. release = single_release ,
. write = cifs_security_flags_proc_write ,
} ;
2008-02-12 23:32:36 +03:00
# else
2008-02-18 07:03:58 +03:00
inline void cifs_proc_init ( void )
2008-02-12 23:32:36 +03:00
{
}
2008-02-18 07:03:58 +03:00
inline void cifs_proc_clean ( void )
2008-02-12 23:32:36 +03:00
{
}
# endif /* PROC_FS */