| // SPDX-License-Identifier: GPL-2.0-only |
| /* |
| * Copyright (C) 2013 Red Hat |
| * Author: Rob Clark <robdclark@gmail.com> |
| */ |
| |
| /* For profiling, userspace can: |
| * |
| * tail -f /sys/kernel/debug/dri/<minor>/gpu |
| * |
| * This will enable performance counters/profiling to track the busy time |
| * and any gpu specific performance counters that are supported. |
| */ |
| |
| #ifdef CONFIG_DEBUG_FS |
| |
| #include <linux/debugfs.h> |
| #include <linux/uaccess.h> |
| |
| #include <drm/drm_file.h> |
| |
| #include "msm_drv.h" |
| #include "msm_gpu.h" |
| |
| struct msm_perf_state { |
| struct drm_device *dev; |
| |
| bool open; |
| int cnt; |
| struct mutex read_lock; |
| |
| char buf[256]; |
| int buftot, bufpos; |
| |
| unsigned long next_jiffies; |
| }; |
| |
| #define SAMPLE_TIME (HZ/4) |
| |
| /* wait for next sample time: */ |
| static int wait_sample(struct msm_perf_state *perf) |
| { |
| unsigned long start_jiffies = jiffies; |
| |
| if (time_after(perf->next_jiffies, start_jiffies)) { |
| unsigned long remaining_jiffies = |
| perf->next_jiffies - start_jiffies; |
| int ret = schedule_timeout_interruptible(remaining_jiffies); |
| if (ret > 0) { |
| /* interrupted */ |
| return -ERESTARTSYS; |
| } |
| } |
| perf->next_jiffies += SAMPLE_TIME; |
| return 0; |
| } |
| |
| static int refill_buf(struct msm_perf_state *perf) |
| { |
| struct msm_drm_private *priv = perf->dev->dev_private; |
| struct msm_gpu *gpu = priv->gpu; |
| char *ptr = perf->buf; |
| int rem = sizeof(perf->buf); |
| int i, n; |
| |
| if ((perf->cnt++ % 32) == 0) { |
| /* Header line: */ |
| n = snprintf(ptr, rem, "%%BUSY"); |
| ptr += n; |
| rem -= n; |
| |
| for (i = 0; i < gpu->num_perfcntrs; i++) { |
| const struct msm_gpu_perfcntr *perfcntr = &gpu->perfcntrs[i]; |
| n = snprintf(ptr, rem, "\t%s", perfcntr->name); |
| ptr += n; |
| rem -= n; |
| } |
| } else { |
| /* Sample line: */ |
| uint32_t activetime = 0, totaltime = 0; |
| uint32_t cntrs[5]; |
| uint32_t val; |
| int ret; |
| |
| /* sleep until next sample time: */ |
| ret = wait_sample(perf); |
| if (ret) |
| return ret; |
| |
| ret = msm_gpu_perfcntr_sample(gpu, &activetime, &totaltime, |
| ARRAY_SIZE(cntrs), cntrs); |
| if (ret < 0) |
| return ret; |
| |
| val = totaltime ? 1000 * activetime / totaltime : 0; |
| n = snprintf(ptr, rem, "%3d.%d%%", val / 10, val % 10); |
| ptr += n; |
| rem -= n; |
| |
| for (i = 0; i < ret; i++) { |
| /* cycle counters (I think).. convert to MHz.. */ |
| val = cntrs[i] / 10000; |
| n = snprintf(ptr, rem, "\t%5d.%02d", |
| val / 100, val % 100); |
| ptr += n; |
| rem -= n; |
| } |
| } |
| |
| n = snprintf(ptr, rem, "\n"); |
| ptr += n; |
| rem -= n; |
| |
| perf->bufpos = 0; |
| perf->buftot = ptr - perf->buf; |
| |
| return 0; |
| } |
| |
| static ssize_t perf_read(struct file *file, char __user *buf, |
| size_t sz, loff_t *ppos) |
| { |
| struct msm_perf_state *perf = file->private_data; |
| int n = 0, ret = 0; |
| |
| mutex_lock(&perf->read_lock); |
| |
| if (perf->bufpos >= perf->buftot) { |
| ret = refill_buf(perf); |
| if (ret) |
| goto out; |
| } |
| |
| n = min((int)sz, perf->buftot - perf->bufpos); |
| if (copy_to_user(buf, &perf->buf[perf->bufpos], n)) { |
| ret = -EFAULT; |
| goto out; |
| } |
| |
| perf->bufpos += n; |
| *ppos += n; |
| |
| out: |
| mutex_unlock(&perf->read_lock); |
| if (ret) |
| return ret; |
| return n; |
| } |
| |
| static int perf_open(struct inode *inode, struct file *file) |
| { |
| struct msm_perf_state *perf = inode->i_private; |
| struct drm_device *dev = perf->dev; |
| struct msm_drm_private *priv = dev->dev_private; |
| struct msm_gpu *gpu = priv->gpu; |
| int ret = 0; |
| |
| if (!gpu) |
| return -ENODEV; |
| |
| mutex_lock(&gpu->lock); |
| |
| if (perf->open) { |
| ret = -EBUSY; |
| goto out; |
| } |
| |
| file->private_data = perf; |
| perf->open = true; |
| perf->cnt = 0; |
| perf->buftot = 0; |
| perf->bufpos = 0; |
| msm_gpu_perfcntr_start(gpu); |
| perf->next_jiffies = jiffies + SAMPLE_TIME; |
| |
| out: |
| mutex_unlock(&gpu->lock); |
| return ret; |
| } |
| |
| static int perf_release(struct inode *inode, struct file *file) |
| { |
| struct msm_perf_state *perf = inode->i_private; |
| struct msm_drm_private *priv = perf->dev->dev_private; |
| msm_gpu_perfcntr_stop(priv->gpu); |
| perf->open = false; |
| return 0; |
| } |
| |
| |
| static const struct file_operations perf_debugfs_fops = { |
| .owner = THIS_MODULE, |
| .open = perf_open, |
| .read = perf_read, |
| .llseek = no_llseek, |
| .release = perf_release, |
| }; |
| |
| int msm_perf_debugfs_init(struct drm_minor *minor) |
| { |
| struct msm_drm_private *priv = minor->dev->dev_private; |
| struct msm_perf_state *perf; |
| |
| /* only create on first minor: */ |
| if (priv->perf) |
| return 0; |
| |
| perf = kzalloc(sizeof(*perf), GFP_KERNEL); |
| if (!perf) |
| return -ENOMEM; |
| |
| perf->dev = minor->dev; |
| |
| mutex_init(&perf->read_lock); |
| priv->perf = perf; |
| |
| debugfs_create_file("perf", S_IFREG | S_IRUGO, minor->debugfs_root, |
| perf, &perf_debugfs_fops); |
| return 0; |
| } |
| |
| void msm_perf_debugfs_cleanup(struct msm_drm_private *priv) |
| { |
| struct msm_perf_state *perf = priv->perf; |
| |
| if (!perf) |
| return; |
| |
| priv->perf = NULL; |
| |
| mutex_destroy(&perf->read_lock); |
| |
| kfree(perf); |
| } |
| |
| #endif |