mmc: Add tracepoints of mmc block operations
authorKen Sumrall <ksumrall@android.com>
Thu, 16 May 2013 03:13:13 +0000 (20:13 -0700)
committerJohn Stultz <john.stultz@linaro.org>
Tue, 16 Feb 2016 21:53:28 +0000 (13:53 -0800)
Add tracepoints to record the start and end of each mmc block
operation.  This includes read, write, erase, secure erase,
trim, secure trim1 and secure trim 2, discard and
sanitize commands.

Change-Id: Ic5d1cbdb9adb940d8b1a2a13c73970023575df50
Signed-off-by: Ken Sumrall <ksumrall@android.com>
drivers/mmc/card/block.c
drivers/mmc/core/core.c
include/trace/events/mmc.h [new file with mode: 0644]

index daffad24e8e5e05de7b9b8b397041e51a23906ab..21aa3244029b277b2287307d16929e78b5383bf9 100644 (file)
@@ -36,6 +36,9 @@
 #include <linux/compat.h>
 #include <linux/pm_runtime.h>
 
+#define CREATE_TRACE_POINTS
+#include <trace/events/mmc.h>
+
 #include <linux/mmc/ioctl.h>
 #include <linux/mmc/card.h>
 #include <linux/mmc/host.h>
@@ -443,9 +446,11 @@ static int ioctl_do_sanitize(struct mmc_card *card)
        pr_debug("%s: %s - SANITIZE IN PROGRESS...\n",
                mmc_hostname(card->host), __func__);
 
+       trace_mmc_blk_erase_start(EXT_CSD_SANITIZE_START, 0, 0);
        err = mmc_switch(card, EXT_CSD_CMD_SET_NORMAL,
                                        EXT_CSD_SANITIZE_START, 1,
                                        MMC_SANITIZE_REQ_TIMEOUT);
+       trace_mmc_blk_erase_end(EXT_CSD_SANITIZE_START, 0, 0);
 
        if (err)
                pr_err("%s: %s - EXT_CSD_SANITIZE_START failed. err=%d\n",
index 9c9bd3c906375c59669dbef7f99616ffd316bab1..95fba49d2d42a552c8e2066fbdc4312c0c5b704b 100644 (file)
@@ -31,6 +31,8 @@
 #include <linux/of.h>
 #include <linux/wakelock.h>
 
+#include <trace/events/mmc.h>
+
 #include <linux/mmc/card.h>
 #include <linux/mmc/host.h>
 #include <linux/mmc/mmc.h>
@@ -178,6 +180,7 @@ void mmc_request_done(struct mmc_host *host, struct mmc_request *mrq)
                        pr_debug("%s:     %d bytes transferred: %d\n",
                                mmc_hostname(host),
                                mrq->data->bytes_xfered, mrq->data->error);
+                       trace_mmc_blk_rw_end(cmd->opcode, cmd->arg, mrq->data);
                }
 
                if (mrq->stop) {
@@ -620,8 +623,12 @@ struct mmc_async_req *mmc_start_req(struct mmc_host *host,
                }
        }
 
-       if (!err && areq)
+       if (!err && areq) {
+               trace_mmc_blk_rw_start(areq->mrq->cmd->opcode,
+                                      areq->mrq->cmd->arg,
+                                      areq->mrq->data);
                start_err = __mmc_start_data_req(host, areq->mrq);
+       }
 
        if (host->areq)
                mmc_post_req(host, host->areq->mrq, 0);
@@ -2058,8 +2065,13 @@ static int mmc_do_erase(struct mmc_card *card, unsigned int from,
        struct mmc_command cmd = {0};
        unsigned int qty = 0;
        unsigned long timeout;
+       unsigned int fr, nr;
        int err;
 
+       fr = from;
+       nr = to - from + 1;
+       trace_mmc_blk_erase_start(arg, fr, nr);
+
        mmc_retune_hold(card->host);
 
        /*
@@ -2166,6 +2178,7 @@ static int mmc_do_erase(struct mmc_card *card, unsigned int from,
                 (R1_CURRENT_STATE(cmd.resp[0]) == R1_STATE_PRG));
 out:
        mmc_retune_release(card->host);
+       trace_mmc_blk_erase_end(arg, fr, nr);
        return err;
 }
 
diff --git a/include/trace/events/mmc.h b/include/trace/events/mmc.h
new file mode 100644 (file)
index 0000000..82b368d
--- /dev/null
@@ -0,0 +1,91 @@
+/*
+ * Copyright (C) 2013 Google, Inc.
+ *
+ * This software is licensed under the terms of the GNU General Public
+ * License version 2, as published by the Free Software Foundation, and
+ * may be copied, distributed, and modified under those terms.
+ *
+ * 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.
+ *
+ */
+
+#undef TRACE_SYSTEM
+#define TRACE_SYSTEM mmc
+
+#if !defined(_TRACE_MMC_H) || defined(TRACE_HEADER_MULTI_READ)
+#define _TRACE_MMC_H
+
+#include <linux/tracepoint.h>
+#include <linux/mmc/mmc.h>
+#include <linux/mmc/core.h>
+
+/*
+ * Unconditional logging of mmc block erase operations,
+ * including cmd, address, size
+ */
+DECLARE_EVENT_CLASS(mmc_blk_erase_class,
+       TP_PROTO(unsigned int cmd, unsigned int addr, unsigned int size),
+       TP_ARGS(cmd, addr, size),
+       TP_STRUCT__entry(
+               __field(unsigned int, cmd)
+               __field(unsigned int, addr)
+               __field(unsigned int, size)
+       ),
+       TP_fast_assign(
+               __entry->cmd = cmd;
+               __entry->addr = addr;
+               __entry->size = size;
+       ),
+       TP_printk("cmd=%u,addr=0x%08x,size=0x%08x",
+                 __entry->cmd, __entry->addr, __entry->size)
+);
+
+DEFINE_EVENT(mmc_blk_erase_class, mmc_blk_erase_start,
+       TP_PROTO(unsigned int cmd, unsigned int addr, unsigned int size),
+       TP_ARGS(cmd, addr, size));
+
+DEFINE_EVENT(mmc_blk_erase_class, mmc_blk_erase_end,
+       TP_PROTO(unsigned int cmd, unsigned int addr, unsigned int size),
+       TP_ARGS(cmd, addr, size));
+
+/*
+ * Logging of start of read or write mmc block operation,
+ * including cmd, address, size
+ */
+DECLARE_EVENT_CLASS(mmc_blk_rw_class,
+       TP_PROTO(unsigned int cmd, unsigned int addr, struct mmc_data *data),
+       TP_ARGS(cmd, addr, data),
+       TP_STRUCT__entry(
+               __field(unsigned int, cmd)
+               __field(unsigned int, addr)
+               __field(unsigned int, size)
+       ),
+       TP_fast_assign(
+               __entry->cmd = cmd;
+               __entry->addr = addr;
+               __entry->size = data->blocks;
+       ),
+       TP_printk("cmd=%u,addr=0x%08x,size=0x%08x",
+                 __entry->cmd, __entry->addr, __entry->size)
+);
+
+DEFINE_EVENT_CONDITION(mmc_blk_rw_class, mmc_blk_rw_start,
+       TP_PROTO(unsigned int cmd, unsigned int addr, struct mmc_data *data),
+       TP_ARGS(cmd, addr, data),
+       TP_CONDITION(((cmd == MMC_READ_MULTIPLE_BLOCK) ||
+                     (cmd == MMC_WRITE_MULTIPLE_BLOCK)) &&
+                     data));
+
+DEFINE_EVENT_CONDITION(mmc_blk_rw_class, mmc_blk_rw_end,
+       TP_PROTO(unsigned int cmd, unsigned int addr, struct mmc_data *data),
+       TP_ARGS(cmd, addr, data),
+       TP_CONDITION(((cmd == MMC_READ_MULTIPLE_BLOCK) ||
+                     (cmd == MMC_WRITE_MULTIPLE_BLOCK)) &&
+                     data));
+#endif /* _TRACE_MMC_H */
+
+/* This part must be outside protection */
+#include <trace/define_trace.h>