All of lore.kernel.org
 help / color / mirror / Atom feed
From: Hannes Reinecke <hare@suse.de>
To: dgilbert@interlog.com, James Bottomley <jbottomley@parallels.com>
Cc: Christoph Hellwig <hch@lst.de>, Robert Elliott <elliott@hp.com>,
	Ewan Milne <emilne@redhat.com>,
	linux-scsi@vger.kernel.org
Subject: Re: [PATCH 03/10] scsi: Implement per-cpu logging buffer
Date: Tue, 18 Nov 2014 16:25:01 +0100	[thread overview]
Message-ID: <546B64CD.4010604@suse.de> (raw)
In-Reply-To: <546A5021.20302@interlog.com>

On 11/17/2014 08:44 PM, Douglas Gilbert wrote:
> On 14-11-17 08:25 AM, Hannes Reinecke wrote:
>> Implement a per-cpu buffer for formatting messages to
>> avoid line break up under high load.
>> This patch implements scmd_printk() and sdev_prefix_printk()
>> using the per-cpu buffer and makes sdev_printk() a wrapper
>> for sdev_prefix_printk().
>>
>> Tested-by: Robert Elliott <elliott@hp.com>
>> Reviewed-by: Robert Elliott <elliott@hp.com>
>> Signed-off-by: Hannes Reinecke <hare@suse.de>
>> ---
>>   drivers/scsi/Makefile       |   2 +-
>>   drivers/scsi/scsi_logging.c | 124
>> ++++++++++++++++++++++++++++++++++++++++++++
>>   include/scsi/scsi_device.h  |  21 +++-----
>>   3 files changed, 133 insertions(+), 14 deletions(-)
>>   create mode 100644 drivers/scsi/scsi_logging.c
>>
>> diff --git a/drivers/scsi/Makefile b/drivers/scsi/Makefile
>> index 59f1ce6..4991b62 100644
>> --- a/drivers/scsi/Makefile
>> +++ b/drivers/scsi/Makefile
>> @@ -166,7 +166,7 @@ scsi_mod-y            += scsi_scan.o
>> scsi_sysfs.o scsi_devinfo.o
>>   scsi_mod-$(CONFIG_SCSI_NETLINK)    += scsi_netlink.o
>>   scsi_mod-$(CONFIG_SYSCTL)    += scsi_sysctl.o
>>   scsi_mod-$(CONFIG_SCSI_PROC_FS)    += scsi_proc.o
>> -scsi_mod-y            += scsi_trace.o
>> +scsi_mod-y            += scsi_trace.o scsi_logging.o
>>   scsi_mod-$(CONFIG_PM)        += scsi_pm.o
>>
>>   hv_storvsc-y            := storvsc_drv.o
>> diff --git a/drivers/scsi/scsi_logging.c
>> b/drivers/scsi/scsi_logging.c
>> new file mode 100644
>> index 0000000..09d65de
>> --- /dev/null
>> +++ b/drivers/scsi/scsi_logging.c
>> @@ -0,0 +1,124 @@
>> +/*
>> + * scsi_logging.c
>> + *
>> + * Copyright (C) 2014 SUSE Linux Products GmbH
>> + * Copyright (C) 2014 Hannes Reinecke <hare@suse.de>
>> + *
>> + * This file is released under the GPLv2
>> + */
>> +
>> +#include <linux/kernel.h>
>> +#include <linux/atomic.h>
>> +
>> +#include <scsi/scsi.h>
>> +#include <scsi/scsi_cmnd.h>
>> +#include <scsi/scsi_device.h>
>> +#include <scsi/scsi_dbg.h>
>> +
>> +#define SCSI_LOG_SPOOLSIZE 4096
>> +#define SCSI_LOG_BUFSIZE 128
>> +
>> +#if (SCSI_LOG_SPOOLSIZE / SCSI_LOG_BUFSIZE) > BITS_PER_LONG
>> +#warning SCSI logging bitmask too large
>> +#endif
>> +
>> +struct scsi_log_buf {
>> +    char buffer[SCSI_LOG_SPOOLSIZE];
>> +    unsigned long map;
>> +};
>> +
>> +static DEFINE_PER_CPU(struct scsi_log_buf, scsi_format_log);
>> +
>> +static char *scsi_log_reserve_buffer(size_t *len)
>> +{
>> +    struct scsi_log_buf *buf;
>> +    unsigned long map_bits = sizeof(buf->buffer) / SCSI_LOG_BUFSIZE;
>> +    unsigned long idx = 0;
>> +
>> +    preempt_disable();
>> +    buf = this_cpu_ptr(&scsi_format_log);
>> +    idx = find_first_zero_bit(&buf->map, map_bits);
>> +    if (likely(idx < map_bits)) {
>> +        while (test_and_set_bit(idx, &buf->map)) {
>> +            idx = find_next_zero_bit(&buf->map, map_bits, idx);
>> +            if (idx >= map_bits)
>> +                break;
>> +        }
>> +    }
>> +    if (WARN_ON(idx >= map_bits)) {
>> +        preempt_enable();
>> +        return NULL;
>> +    }
>> +    *len = SCSI_LOG_BUFSIZE;
>> +    return buf->buffer + idx * SCSI_LOG_BUFSIZE;
>> +}
>> +
>> +static void scsi_log_release_buffer(char *bufptr)
>> +{
>> +    struct scsi_log_buf *buf;
>> +    unsigned long idx;
>> +    int ret;
>> +
>> +    buf = this_cpu_ptr(&scsi_format_log);
>> +    if (bufptr >= buf->buffer &&
>> +        bufptr < buf->buffer + SCSI_LOG_SPOOLSIZE) {
>> +        idx = (bufptr - buf->buffer) / SCSI_LOG_BUFSIZE;
>> +        ret = test_and_clear_bit(idx, &buf->map);
>> +        WARN_ON(!ret);
>> +    }
>> +    preempt_enable();
>> +}
>> +
>> +int sdev_prefix_printk(const char *level, const struct
>> scsi_device *sdev,
>> +               const char *name, const char *fmt, ...)
>> +{
>> +    va_list args;
>> +    char *logbuf;
>> +    size_t off = 0, logbuf_len;
>> +    int ret;
>> +
>> +    if (!sdev)
>> +        return 0;
>> +
>> +    logbuf = scsi_log_reserve_buffer(&logbuf_len);
>> +    if (!logbuf)
>> +        return 0;
>> +
>> +    if (name)
>> +        off += scnprintf(logbuf + off, logbuf_len - off,
>> +                 "[%s] ", name);
>> +    va_start(args, fmt);
>> +    off += vscnprintf(logbuf + off, logbuf_len - off, fmt, args);
>> +    va_end(args);
>> +    ret = dev_printk(level, &sdev->sdev_gendev, "%s", logbuf);
>> +    scsi_log_release_buffer(logbuf);
>> +    return ret;
>> +}
>> +EXPORT_SYMBOL(sdev_prefix_printk);
>> +
>> +int scmd_printk(const char *level, const struct scsi_cmnd *scmd,
>> +        const char *fmt, ...)
>> +{
>> +    struct gendisk *disk = scmd->request->rq_disk;
>> +    va_list args;
>> +    char *logbuf;
>> +    size_t off = 0, logbuf_len;
>> +    int ret;
>> +
>> +    if (!scmd || !scmd->cmnd)
>> +        return 0;
>> +
>> +    logbuf = scsi_log_reserve_buffer(&logbuf_len);
>> +    if (!logbuf)
>> +        return 0;
>> +    if (disk)
>> +        off += scnprintf(logbuf + off, logbuf_len - off,
>> +                 "[%s] ", disk->disk_name);
>> +    va_start(args, fmt);
>> +    off += vscnprintf(logbuf + off, logbuf_len - off, fmt, args);
>> +    va_end(args);
>> +    ret = dev_printk(level, &scmd->device->sdev_gendev, "%s",
>> logbuf);
>> +    scsi_log_release_buffer(logbuf);
>> +    return ret;
>> +}
>> +EXPORT_SYMBOL(scmd_printk);
>> diff --git a/include/scsi/scsi_device.h b/include/scsi/scsi_device.h
>> index 0aeaa00..6fddcd2 100644
>> --- a/include/scsi/scsi_device.h
>> +++ b/include/scsi/scsi_device.h
>> @@ -230,9 +230,6 @@ struct scsi_dh_data {
>>   #define transport_class_to_sdev(class_dev) \
>>       to_scsi_device(class_dev->parent)
>>
>> -#define sdev_printk(prefix, sdev, fmt, a...)    \
>> -    dev_printk(prefix, &(sdev)->sdev_gendev, fmt, ##a)
>> -
>>   #define sdev_dbg(sdev, fmt, a...) \
>>       dev_dbg(&(sdev)->sdev_gendev, fmt, ##a)
>>
>> @@ -240,16 +237,14 @@ struct scsi_dh_data {
>>    * like scmd_printk, but the device name is passed in
>>    * as a string pointer
>>    */
>> -#define sdev_prefix_printk(l, sdev, p, fmt, a...)            \
>> -    (p) ?                                \
>> -    sdev_printk(l, sdev, "[%s] " fmt, p, ##a) :            \
>> -    sdev_printk(l, sdev, fmt, ##a)
>> -
>> -#define scmd_printk(prefix, scmd, fmt, a...)                \
>> -        (scmd)->request->rq_disk ?                    \
>> -    sdev_printk(prefix, (scmd)->device, "[%s] " fmt,        \
>> -            (scmd)->request->rq_disk->disk_name, ##a) :        \
>> -    sdev_printk(prefix, (scmd)->device, fmt, ##a)
>> +extern int sdev_prefix_printk(const char *, const struct
>> scsi_device *,
>> +                  const char *, const char *, ...);
>> +
>> +#define sdev_printk(l, sdev, fmt, a...)                \
>> +    sdev_prefix_printk(l, sdev, NULL, fmt, ##a)
>> +
>> +extern int scmd_printk(const char *, const struct scsi_cmnd *,
>> +               const char *, ...);
>>
>>   #define scmd_dbg(scmd, fmt, a...)                       \
>>       do {                                   \
>>
> 
> The gcc compiler knows about the idiosyncrasies of printf()
> and will warn about abuses like this:
>    int k;
>    printf("string=%s\n", k);
> 
> 
> In sg3_utils I sometimes collapse fprintf(stderr, const char fmt ...)
> with a new function declaration like this:
>    int pr2serr(const char * fmt, ...)
>         __attribute__ ((format (printf, 1, 2)));
> 
> The "1" is the position of the fmt string (origin 1) and the
> "2" is the position of the first arg to check. That __attribute__
> tells the compiler how to do printf() style checking on
> pr2serr().
> 
> 
> In the kernel include compiler-gcc.h contains this define:
> 
>   #define __printf(a, b)   __attribute__((format(printf, a, b)))
> 
> So I would suggest changing:
>   extern int sdev_prefix_printk(const char *, const struct
> scsi_device *,
>                       const char *, const char *, ...);
> to
>   extern __printf(4, 5) int
>      sdev_prefix_printk(const char *, const struct scsi_device *,
>                       const char *, const char *, ...);
> 
> 
> plus a similar change to the scmd_printk() declaration (and any
> other printf() style functions being introduced).
> 
Hmm. I guess you are right. Will be updating the patch.

Cheers,

Hannes
-- 
Dr. Hannes Reinecke		      zSeries & Storage
hare@suse.de			      +49 911 74053 688
SUSE LINUX GmbH, Maxfeldstr. 5, 90409 Nürnberg
GF: J. Hawn, J. Guild, F. Imendörffer, HRB 21284 (AG Nürnberg)
--
To unsubscribe from this list: send the line "unsubscribe linux-scsi" in
the body of a message to majordomo@vger.kernel.org
More majordomo info at  http://vger.kernel.org/majordomo-info.html

  reply	other threads:[~2014-11-18 15:25 UTC|newest]

Thread overview: 17+ messages / expand[flat|nested]  mbox.gz  Atom feed  top
2014-11-17 13:25 [PATCHv3 00/10] scsi logging update: the real thing Hannes Reinecke
2014-11-17 13:25 ` [PATCH 01/10] scsi: Rename SERVICE_ACTION_IN to SERVICE_ACTION_IN_16 Hannes Reinecke
2014-11-17 13:25 ` [PATCH 02/10] scsi: Add SPC-3 command definitions Hannes Reinecke
2014-11-17 13:25 ` [PATCH 03/10] scsi: Implement per-cpu logging buffer Hannes Reinecke
2014-11-17 19:44   ` Douglas Gilbert
2014-11-18 15:25     ` Hannes Reinecke [this message]
2014-11-17 13:25 ` [PATCH 04/10] scsi: log request tag for scmd_printk() Hannes Reinecke
2014-11-17 13:25 ` [PATCH 05/10] scsi: use external buffer for command logging Hannes Reinecke
2014-11-17 13:25 ` [PATCH 06/10] libata: use __scsi_format_command() Hannes Reinecke
2014-11-17 13:25 ` [PATCH 07/10] scsi: use per-cpu buffer for formatting sense Hannes Reinecke
2014-11-17 13:25 ` [PATCH 08/10] scsi: use per-cpu buffer for formatting scsi_print_result() Hannes Reinecke
2014-11-17 13:25 ` [PATCH 09/10] scsi: Conditionally compile in constants.c Hannes Reinecke
2014-11-17 13:25 ` [PATCH 10/10] scsi: Do not display buffer pointers in scsi_log_send() Hannes Reinecke
2015-01-05 16:59 ` [PATCHv3 00/10] scsi logging update: the real thing Christoph Hellwig
2015-01-06 18:26   ` Hannes Reinecke
  -- strict thread matches above, loose matches on Subject: below --
2014-11-06  8:30 [PATCHv2 " Hannes Reinecke
2014-11-06  8:30 ` [PATCH 03/10] scsi: Implement per-cpu logging buffer Hannes Reinecke
2014-11-13  2:22   ` Elliott, Robert (Server Storage)

Reply instructions:

You may reply publicly to this message via plain-text email
using any one of the following methods:

* Save the following mbox file, import it into your mail client,
  and reply-to-all from there: mbox

  Avoid top-posting and favor interleaved quoting:
  https://en.wikipedia.org/wiki/Posting_style#Interleaved_style

* Reply using the --to, --cc, and --in-reply-to
  switches of git-send-email(1):

  git send-email \
    --in-reply-to=546B64CD.4010604@suse.de \
    --to=hare@suse.de \
    --cc=dgilbert@interlog.com \
    --cc=elliott@hp.com \
    --cc=emilne@redhat.com \
    --cc=hch@lst.de \
    --cc=jbottomley@parallels.com \
    --cc=linux-scsi@vger.kernel.org \
    /path/to/YOUR_REPLY

  https://kernel.org/pub/software/scm/git/docs/git-send-email.html

* If your mail client supports setting the In-Reply-To header
  via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line before the message body.
This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.