On Fri, Oct 28, 2011 at 11:48 PM, Jeff Layton <[email protected]> wrote:
> On Fri, 28 Oct 2011 23:54:13 +0400
> Pavel Shilovsky <[email protected]> wrote:
>
>> From: Steve French <[email protected]>
>>
>> Adding SMB2 statistics requires changes to the way cifs handles stats.
>> Since there are only 19 command codes, it also is easier to track by exact
>> command code than it was for cifs. Turn the counters for protocol
>> ops sent to be a union (one struct for cifs, one for smb2). While at it
>> split out the functions which clear stats and prints stats into their own
>> subfunctions so they are easy to read and don't go past 80 columns.
>>
>> Signed-off-by: Steve French <[email protected]>
>> Signed-off-by: Pavel Shilovsky <[email protected]>
>> ---
>> fs/cifs/cifs_debug.c | 146
>> ++++++++++++++++++++++++++++++-------------------
>> fs/cifs/cifsglob.h | 56 ++++++++++++-------
>> fs/cifs/cifssmb.c | 54 +++++++++---------
>> fs/cifs/misc.c | 2 +-
>> 4 files changed, 152 insertions(+), 106 deletions(-)
>>
>> diff --git a/fs/cifs/cifs_debug.c b/fs/cifs/cifs_debug.c
>> index 84e8c07..8ccdb15 100644
>> --- a/fs/cifs/cifs_debug.c
>> +++ b/fs/cifs/cifs_debug.c
>> @@ -249,6 +249,55 @@ static const struct file_operations
>> cifs_debug_data_proc_fops = {
>> };
>>
>> #ifdef CONFIG_CIFS_STATS
>> +
>> +#ifdef CONFIG_CIFS_SMB2
>> +static void smb2_clear_stats(struct cifs_tcon *tcon)
>> +{
>> + int i;
>> +
>> + atomic_set(&tcon->num_smbs_sent, 0);
>> + for (i = 0; i < NUMBER_OF_SMB2_COMMANDS; i++) {
>> + atomic_set(&tcon->stats.smb2_stats.smb2_com_sent[i], 0);
>> + atomic_set(&tcon->stats.smb2_stats.smb2_com_fail[i], 0);
>> + }
>> +}
>> +#endif /* CONFIG_CIFS_SMB2 */
>> +
>> +static void clear_cifs_stats(struct cifs_tcon *tcon)
>> +{
>> + atomic_set(&tcon->num_smbs_sent, 0);
>> +
>> +#ifdef CONFIG_CIFS_SMB2
>> + if (tcon->ses->server->is_smb2) {
>> + smb2_clear_stats(tcon);
>> + return;
>> + }
>> +#endif /* CONFIG_CIFS_SMB2 */
>> +
>
> ^^^^^^^^^^^^^
> The above logic should be encapsulated in smb2_clear_stats, and that
> function should just be a noop when CONFIG_CIFS_SMB2 is not set.
makes sense - could be easier to read your way
>> + /* cifs specific statistics, not applicable to smb2 sessions */
>> + atomic_set(&tcon->stats.cifs_stats.num_writes, 0);
>> + atomic_set(&tcon->stats.cifs_stats.num_reads, 0);
>> + atomic_set(&tcon->stats.cifs_stats.num_flushes, 0);
>> + atomic_set(&tcon->stats.cifs_stats.num_oplock_brks, 0);
>> + atomic_set(&tcon->stats.cifs_stats.num_opens, 0);
>> + atomic_set(&tcon->stats.cifs_stats.num_posixopens, 0);
>> + atomic_set(&tcon->stats.cifs_stats.num_posixmkdirs, 0);
>> + atomic_set(&tcon->stats.cifs_stats.num_closes, 0);
>> + atomic_set(&tcon->stats.cifs_stats.num_deletes, 0);
>> + atomic_set(&tcon->stats.cifs_stats.num_mkdirs, 0);
>> + atomic_set(&tcon->stats.cifs_stats.num_rmdirs, 0);
>> + atomic_set(&tcon->stats.cifs_stats.num_renames, 0);
>> + atomic_set(&tcon->stats.cifs_stats.num_t2renames, 0);
>> + atomic_set(&tcon->stats.cifs_stats.num_ffirst, 0);
>> + atomic_set(&tcon->stats.cifs_stats.num_fnext, 0);
>> + atomic_set(&tcon->stats.cifs_stats.num_fclose, 0);
>> + atomic_set(&tcon->stats.cifs_stats.num_hardlinks, 0);
>> + atomic_set(&tcon->stats.cifs_stats.num_symlinks, 0);
>> + atomic_set(&tcon->stats.cifs_stats.num_locks, 0);
>> + atomic_set(&tcon->stats.cifs_stats.num_acl_get, 0);
>> + atomic_set(&tcon->stats.cifs_stats.num_acl_set, 0);
>> +}
>> +
>> static ssize_t cifs_stats_proc_write(struct file *file,
>> const char __user *buffer, size_t count, loff_t *ppos)
>> {
>> @@ -279,25 +328,7 @@ static ssize_t cifs_stats_proc_write(struct file *file,
>> tcon = list_entry(tmp3,
>> struct cifs_tcon,
>> 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);
>> - atomic_set(&tcon->num_posixopens, 0);
>> - atomic_set(&tcon->num_posixmkdirs, 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);
>> + clear_cifs_stats(tcon);
>> }
>> }
>> }
>> @@ -307,6 +338,44 @@ static ssize_t cifs_stats_proc_write(struct file *file,
>> return count;
>> }
>>
>> +static void cifs_stats_print(struct seq_file *m, struct cifs_tcon *tcon)
>> +{
>> + if (tcon->need_reconnect)
>> + seq_puts(m, "\tDISCONNECTED ");
>> + seq_printf(m, "\nSMBs: %d Oplock Breaks: %d",
>> + atomic_read(&tcon->num_smbs_sent),
>> + atomic_read(&tcon->stats.cifs_stats.num_oplock_brks));
>> + seq_printf(m, "\nReads: %d Bytes: %lld",
>> + atomic_read(&tcon->stats.cifs_stats.num_reads),
>> + (long long)(tcon->bytes_read));
>> + seq_printf(m, "\nWrites: %d Bytes: %lld",
>> + atomic_read(&tcon->stats.cifs_stats.num_writes),
>> + (long long)(tcon->bytes_written));
>> + seq_printf(m, "\nFlushes: %d",
>> + atomic_read(&tcon->stats.cifs_stats.num_flushes));
>> + seq_printf(m, "\nLocks: %d HardLinks: %d Symlinks: %d",
>> + atomic_read(&tcon->stats.cifs_stats.num_locks),
>> + atomic_read(&tcon->stats.cifs_stats.num_hardlinks),
>> + atomic_read(&tcon->stats.cifs_stats.num_symlinks));
>> + seq_printf(m, "\nOpens: %d Closes: %d Deletes: %d",
>> + atomic_read(&tcon->stats.cifs_stats.num_opens),
>> + atomic_read(&tcon->stats.cifs_stats.num_closes),
>> + atomic_read(&tcon->stats.cifs_stats.num_deletes));
>> + seq_printf(m, "\nPosix Opens: %d Posix Mkdirs: %d",
>> + atomic_read(&tcon->stats.cifs_stats.num_posixopens),
>> + atomic_read(&tcon->stats.cifs_stats.num_posixmkdirs));
>> + seq_printf(m, "\nMkdirs: %d Rmdirs: %d",
>> + atomic_read(&tcon->stats.cifs_stats.num_mkdirs),
>> + atomic_read(&tcon->stats.cifs_stats.num_rmdirs));
>> + seq_printf(m, "\nRenames: %d T2 Renames %d",
>> + atomic_read(&tcon->stats.cifs_stats.num_renames),
>> + atomic_read(&tcon->stats.cifs_stats.num_t2renames));
>> + seq_printf(m, "\nFindFirst: %d FNext %d FClose %d",
>> + atomic_read(&tcon->stats.cifs_stats.num_ffirst),
>> + atomic_read(&tcon->stats.cifs_stats.num_fnext),
>> + atomic_read(&tcon->stats.cifs_stats.num_fclose));
>> +}
>> +
>> static int cifs_stats_proc_show(struct seq_file *m, void *v)
>> {
>> int i;
>> @@ -354,44 +423,7 @@ static int cifs_stats_proc_show(struct seq_file *m,
>> void *v)
>> tcon_list);
>> i++;
>> seq_printf(m, "\n%d) %s", i, tcon->treeName);
>> - if (tcon->need_reconnect)
>> - seq_puts(m, "\tDISCONNECTED ");
>> - seq_printf(m, "\nSMBs: %d Oplock Breaks: %d",
>> - atomic_read(&tcon->num_smbs_sent),
>> - atomic_read(&tcon->num_oplock_brks));
>> - seq_printf(m, "\nReads: %d Bytes: %lld",
>> - atomic_read(&tcon->num_reads),
>> - (long long)(tcon->bytes_read));
>> - seq_printf(m, "\nWrites: %d Bytes: %lld",
>> - atomic_read(&tcon->num_writes),
>> - (long long)(tcon->bytes_written));
>> - seq_printf(m, "\nFlushes: %d",
>> - atomic_read(&tcon->num_flushes));
>> - seq_printf(m, "\nLocks: %d HardLinks: %d "
>> - "Symlinks: %d",
>> - atomic_read(&tcon->num_locks),
>> - atomic_read(&tcon->num_hardlinks),
>> - atomic_read(&tcon->num_symlinks));
>> - seq_printf(m, "\nOpens: %d Closes: %d "
>> - "Deletes: %d",
>> - atomic_read(&tcon->num_opens),
>> - atomic_read(&tcon->num_closes),
>> - atomic_read(&tcon->num_deletes));
>> - seq_printf(m, "\nPosix Opens: %d "
>> - "Posix Mkdirs: %d",
>> - atomic_read(&tcon->num_posixopens),
>> - atomic_read(&tcon->num_posixmkdirs));
>> - seq_printf(m, "\nMkdirs: %d Rmdirs: %d",
>> - atomic_read(&tcon->num_mkdirs),
>> - atomic_read(&tcon->num_rmdirs));
>> - seq_printf(m, "\nRenames: %d T2 Renames %d",
>> - atomic_read(&tcon->num_renames),
>> - atomic_read(&tcon->num_t2renames));
>> - seq_printf(m, "\nFindFirst: %d FNext %d "
>> - "FClose %d",
>> - atomic_read(&tcon->num_ffirst),
>> - atomic_read(&tcon->num_fnext),
>> - atomic_read(&tcon->num_fclose));
>> + cifs_stats_print(m, tcon);
>> }
>> }
>> }
>> diff --git a/fs/cifs/cifsglob.h b/fs/cifs/cifsglob.h
>> index 4c38b04..6dfc7ef 100644
>> --- a/fs/cifs/cifsglob.h
>> +++ b/fs/cifs/cifsglob.h
>> @@ -292,6 +292,7 @@ struct TCP_Server_Info {
>> bool sec_kerberos; /* supports plain Kerberos */
>> bool sec_mskerberos; /* supports legacy MS Kerberos */
>> bool large_buf; /* is current buffer large? */
>> + bool is_smb2; /* smb2 not cifs protocol negotiated */
>> struct delayed_work echo; /* echo ping workqueue job */
>> struct kvec *iov; /* reusable kvec array for receives */
>> unsigned int nr_iov; /* number of kvecs in array */
>> @@ -392,6 +393,9 @@ struct cifs_ses {
>> negotiate one of the older LANMAN dialects */
>> #define CIFS_SES_LANMAN 8
>> #define CIFS_SES_SMB2 16
>> +
>> +#define NUMBER_OF_SMB2_COMMANDS 0x0013
>> +
>> /*
>> * there is one of these for each connection to a resource on a particular
>> * session
>> @@ -409,27 +413,37 @@ struct cifs_tcon {
>> enum statusEnum tidStatus;
>> #ifdef CONFIG_CIFS_STATS
>> atomic_t num_smbs_sent;
>> - atomic_t num_writes;
>> - atomic_t num_reads;
>> - atomic_t num_flushes;
>> - atomic_t num_oplock_brks;
>> - atomic_t num_opens;
>> - atomic_t num_closes;
>> - atomic_t num_deletes;
>> - atomic_t num_mkdirs;
>> - atomic_t num_posixopens;
>> - atomic_t num_posixmkdirs;
>> - atomic_t num_rmdirs;
>> - atomic_t num_renames;
>> - atomic_t num_t2renames;
>> - atomic_t num_ffirst;
>> - atomic_t num_fnext;
>> - atomic_t num_fclose;
>> - atomic_t num_hardlinks;
>> - atomic_t num_symlinks;
>> - atomic_t num_locks;
>> - atomic_t num_acl_get;
>> - atomic_t num_acl_set;
>> + union {
>> + struct {
>> + atomic_t num_writes;
>> + atomic_t num_reads;
>> + atomic_t num_flushes;
>> + atomic_t num_oplock_brks;
>> + atomic_t num_opens;
>> + atomic_t num_closes;
>> + atomic_t num_deletes;
>> + atomic_t num_mkdirs;
>> + atomic_t num_posixopens;
>> + atomic_t num_posixmkdirs;
>> + atomic_t num_rmdirs;
>> + atomic_t num_renames;
>> + atomic_t num_t2renames;
>> + atomic_t num_ffirst;
>> + atomic_t num_fnext;
>> + atomic_t num_fclose;
>> + atomic_t num_hardlinks;
>> + atomic_t num_symlinks;
>> + atomic_t num_locks;
>> + atomic_t num_acl_get;
>> + atomic_t num_acl_set;
>> + } cifs_stats;
>> +#ifdef CONFIG_CIFS_SMB2
>> + struct {
>> + atomic_t smb2_com_sent[NUMBER_OF_SMB2_COMMANDS];
>> + atomic_t smb2_com_fail[NUMBER_OF_SMB2_COMMANDS];
>> + } smb2_stats;
>
> Is it really necessary to do this with atomics? Those can have
> significant performance impact (TLB flushes, and we don't seem to need the
> guarantees that they provide for simple counters like this. Perhaps
> these should be switched to per-cpu variables or just plain ints?
presumably there is precedent for use of atomics rather than
wrapping updates to these in a mutex - if they are just int
would we have a case where the two overlapping updates could
badly corrupt the value?
--
Thanks,
Steve
--
To unsubscribe from this list: send the line "unsubscribe linux-cifs" in
the body of a message to [email protected]
More majordomo info at http://vger.kernel.org/majordomo-info.html