./drivers/gpu/drm/imagination/pvr_fw_trace.c:251:2-3: Unneeded semicolon Reported-by: Abaci Robot <abaci@linux.alibaba.com> Closes: https://bugzilla.openanolis.cn/show_bug.cgi?id=7694 Signed-off-by: Yang Li <yang.lee@linux.alibaba.com> Signed-off-by: Maxime Ripard <mripard@kernel.org> Link: https://patchwork.freedesktop.org/patch/msgid/20231208003034.68339-1-yang.lee@linux.alibaba.com
472 lines
13 KiB
C
472 lines
13 KiB
C
// SPDX-License-Identifier: GPL-2.0-only OR MIT
|
|
/* Copyright (c) 2023 Imagination Technologies Ltd. */
|
|
|
|
#include "pvr_device.h"
|
|
#include "pvr_gem.h"
|
|
#include "pvr_rogue_fwif.h"
|
|
#include "pvr_rogue_fwif_sf.h"
|
|
#include "pvr_fw_trace.h"
|
|
|
|
#include <drm/drm_drv.h>
|
|
#include <drm/drm_file.h>
|
|
|
|
#include <linux/build_bug.h>
|
|
#include <linux/dcache.h>
|
|
#include <linux/sysfs.h>
|
|
#include <linux/types.h>
|
|
|
|
static void
|
|
tracebuf_ctrl_init(void *cpu_ptr, void *priv)
|
|
{
|
|
struct rogue_fwif_tracebuf *tracebuf_ctrl = cpu_ptr;
|
|
struct pvr_fw_trace *fw_trace = priv;
|
|
u32 thread_nr;
|
|
|
|
tracebuf_ctrl->tracebuf_size_in_dwords = ROGUE_FW_TRACE_BUF_DEFAULT_SIZE_IN_DWORDS;
|
|
tracebuf_ctrl->tracebuf_flags = 0;
|
|
|
|
if (fw_trace->group_mask)
|
|
tracebuf_ctrl->log_type = fw_trace->group_mask | ROGUE_FWIF_LOG_TYPE_TRACE;
|
|
else
|
|
tracebuf_ctrl->log_type = ROGUE_FWIF_LOG_TYPE_NONE;
|
|
|
|
for (thread_nr = 0; thread_nr < ARRAY_SIZE(fw_trace->buffers); thread_nr++) {
|
|
struct rogue_fwif_tracebuf_space *tracebuf_space =
|
|
&tracebuf_ctrl->tracebuf[thread_nr];
|
|
struct pvr_fw_trace_buffer *trace_buffer = &fw_trace->buffers[thread_nr];
|
|
|
|
pvr_fw_object_get_fw_addr(trace_buffer->buf_obj,
|
|
&tracebuf_space->trace_buffer_fw_addr);
|
|
|
|
tracebuf_space->trace_buffer = trace_buffer->buf;
|
|
tracebuf_space->trace_pointer = 0;
|
|
}
|
|
}
|
|
|
|
int pvr_fw_trace_init(struct pvr_device *pvr_dev)
|
|
{
|
|
struct pvr_fw_trace *fw_trace = &pvr_dev->fw_dev.fw_trace;
|
|
struct drm_device *drm_dev = from_pvr_device(pvr_dev);
|
|
u32 thread_nr;
|
|
int err;
|
|
|
|
for (thread_nr = 0; thread_nr < ARRAY_SIZE(fw_trace->buffers); thread_nr++) {
|
|
struct pvr_fw_trace_buffer *trace_buffer = &fw_trace->buffers[thread_nr];
|
|
|
|
trace_buffer->buf =
|
|
pvr_fw_object_create_and_map(pvr_dev,
|
|
ROGUE_FW_TRACE_BUF_DEFAULT_SIZE_IN_DWORDS *
|
|
sizeof(*trace_buffer->buf),
|
|
PVR_BO_FW_FLAGS_DEVICE_UNCACHED |
|
|
PVR_BO_FW_NO_CLEAR_ON_RESET,
|
|
NULL, NULL, &trace_buffer->buf_obj);
|
|
if (IS_ERR(trace_buffer->buf)) {
|
|
drm_err(drm_dev, "Unable to allocate trace buffer\n");
|
|
err = PTR_ERR(trace_buffer->buf);
|
|
trace_buffer->buf = NULL;
|
|
goto err_free_buf;
|
|
}
|
|
}
|
|
|
|
/* TODO: Provide control of group mask. */
|
|
fw_trace->group_mask = 0;
|
|
|
|
fw_trace->tracebuf_ctrl =
|
|
pvr_fw_object_create_and_map(pvr_dev,
|
|
sizeof(*fw_trace->tracebuf_ctrl),
|
|
PVR_BO_FW_FLAGS_DEVICE_UNCACHED |
|
|
PVR_BO_FW_NO_CLEAR_ON_RESET,
|
|
tracebuf_ctrl_init, fw_trace,
|
|
&fw_trace->tracebuf_ctrl_obj);
|
|
if (IS_ERR(fw_trace->tracebuf_ctrl)) {
|
|
drm_err(drm_dev, "Unable to allocate trace buffer control structure\n");
|
|
err = PTR_ERR(fw_trace->tracebuf_ctrl);
|
|
goto err_free_buf;
|
|
}
|
|
|
|
BUILD_BUG_ON(ARRAY_SIZE(fw_trace->tracebuf_ctrl->tracebuf) !=
|
|
ARRAY_SIZE(fw_trace->buffers));
|
|
|
|
for (thread_nr = 0; thread_nr < ARRAY_SIZE(fw_trace->buffers); thread_nr++) {
|
|
struct rogue_fwif_tracebuf_space *tracebuf_space =
|
|
&fw_trace->tracebuf_ctrl->tracebuf[thread_nr];
|
|
struct pvr_fw_trace_buffer *trace_buffer = &fw_trace->buffers[thread_nr];
|
|
|
|
trace_buffer->tracebuf_space = tracebuf_space;
|
|
}
|
|
|
|
return 0;
|
|
|
|
err_free_buf:
|
|
for (thread_nr = 0; thread_nr < ARRAY_SIZE(fw_trace->buffers); thread_nr++) {
|
|
struct pvr_fw_trace_buffer *trace_buffer = &fw_trace->buffers[thread_nr];
|
|
|
|
if (trace_buffer->buf)
|
|
pvr_fw_object_unmap_and_destroy(trace_buffer->buf_obj);
|
|
}
|
|
|
|
return err;
|
|
}
|
|
|
|
void pvr_fw_trace_fini(struct pvr_device *pvr_dev)
|
|
{
|
|
struct pvr_fw_trace *fw_trace = &pvr_dev->fw_dev.fw_trace;
|
|
u32 thread_nr;
|
|
|
|
for (thread_nr = 0; thread_nr < ARRAY_SIZE(fw_trace->buffers); thread_nr++) {
|
|
struct pvr_fw_trace_buffer *trace_buffer = &fw_trace->buffers[thread_nr];
|
|
|
|
pvr_fw_object_unmap_and_destroy(trace_buffer->buf_obj);
|
|
}
|
|
pvr_fw_object_unmap_and_destroy(fw_trace->tracebuf_ctrl_obj);
|
|
}
|
|
|
|
#if defined(CONFIG_DEBUG_FS)
|
|
|
|
/**
|
|
* update_logtype() - Send KCCB command to trigger FW to update logtype
|
|
* @pvr_dev: Target PowerVR device
|
|
* @group_mask: New log group mask.
|
|
*
|
|
* Returns:
|
|
* * 0 on success,
|
|
* * Any error returned by pvr_kccb_send_cmd(), or
|
|
* * -%EIO if the device is lost.
|
|
*/
|
|
static int
|
|
update_logtype(struct pvr_device *pvr_dev, u32 group_mask)
|
|
{
|
|
struct pvr_fw_trace *fw_trace = &pvr_dev->fw_dev.fw_trace;
|
|
struct rogue_fwif_kccb_cmd cmd;
|
|
int idx;
|
|
int err;
|
|
|
|
if (group_mask)
|
|
fw_trace->tracebuf_ctrl->log_type = ROGUE_FWIF_LOG_TYPE_TRACE | group_mask;
|
|
else
|
|
fw_trace->tracebuf_ctrl->log_type = ROGUE_FWIF_LOG_TYPE_NONE;
|
|
|
|
fw_trace->group_mask = group_mask;
|
|
|
|
down_read(&pvr_dev->reset_sem);
|
|
if (!drm_dev_enter(from_pvr_device(pvr_dev), &idx)) {
|
|
err = -EIO;
|
|
goto err_up_read;
|
|
}
|
|
|
|
cmd.cmd_type = ROGUE_FWIF_KCCB_CMD_LOGTYPE_UPDATE;
|
|
cmd.kccb_flags = 0;
|
|
|
|
err = pvr_kccb_send_cmd(pvr_dev, &cmd, NULL);
|
|
|
|
drm_dev_exit(idx);
|
|
|
|
err_up_read:
|
|
up_read(&pvr_dev->reset_sem);
|
|
|
|
return err;
|
|
}
|
|
|
|
struct pvr_fw_trace_seq_data {
|
|
/** @buffer: Pointer to copy of trace data. */
|
|
u32 *buffer;
|
|
|
|
/** @start_offset: Starting offset in trace data, as reported by FW. */
|
|
u32 start_offset;
|
|
|
|
/** @idx: Current index into trace data. */
|
|
u32 idx;
|
|
|
|
/** @assert_buf: Trace assert buffer, as reported by FW. */
|
|
struct rogue_fwif_file_info_buf assert_buf;
|
|
};
|
|
|
|
static u32 find_sfid(u32 id)
|
|
{
|
|
u32 i;
|
|
|
|
for (i = 0; i < ARRAY_SIZE(stid_fmts); i++) {
|
|
if (stid_fmts[i].id == id)
|
|
return i;
|
|
}
|
|
|
|
return ROGUE_FW_SF_LAST;
|
|
}
|
|
|
|
static u32 read_fw_trace(struct pvr_fw_trace_seq_data *trace_seq_data, u32 offset)
|
|
{
|
|
u32 idx;
|
|
|
|
idx = trace_seq_data->idx + offset;
|
|
if (idx >= ROGUE_FW_TRACE_BUF_DEFAULT_SIZE_IN_DWORDS)
|
|
return 0;
|
|
|
|
idx = (idx + trace_seq_data->start_offset) % ROGUE_FW_TRACE_BUF_DEFAULT_SIZE_IN_DWORDS;
|
|
return trace_seq_data->buffer[idx];
|
|
}
|
|
|
|
/**
|
|
* fw_trace_get_next() - Advance trace index to next entry
|
|
* @trace_seq_data: Trace sequence data.
|
|
*
|
|
* Returns:
|
|
* * %true if trace index is now pointing to a valid entry, or
|
|
* * %false if trace index is pointing to an invalid entry, or has hit the end
|
|
* of the trace.
|
|
*/
|
|
static bool fw_trace_get_next(struct pvr_fw_trace_seq_data *trace_seq_data)
|
|
{
|
|
u32 id, sf_id;
|
|
|
|
while (trace_seq_data->idx < ROGUE_FW_TRACE_BUF_DEFAULT_SIZE_IN_DWORDS) {
|
|
id = read_fw_trace(trace_seq_data, 0);
|
|
trace_seq_data->idx++;
|
|
if (!ROGUE_FW_LOG_VALIDID(id))
|
|
continue;
|
|
if (id == ROGUE_FW_SF_MAIN_ASSERT_FAILED) {
|
|
/* Assertion failure marks the end of the trace. */
|
|
return false;
|
|
}
|
|
|
|
sf_id = find_sfid(id);
|
|
if (sf_id == ROGUE_FW_SF_FIRST)
|
|
continue;
|
|
if (sf_id == ROGUE_FW_SF_LAST) {
|
|
/*
|
|
* Could not match with an ID in the SF table, trace is
|
|
* most likely corrupt from this point.
|
|
*/
|
|
return false;
|
|
}
|
|
|
|
/* Skip over the timestamp, and any parameters. */
|
|
trace_seq_data->idx += 2 + ROGUE_FW_SF_PARAMNUM(id);
|
|
|
|
/* Ensure index is now pointing to a valid trace entry. */
|
|
id = read_fw_trace(trace_seq_data, 0);
|
|
if (!ROGUE_FW_LOG_VALIDID(id))
|
|
continue;
|
|
|
|
return true;
|
|
}
|
|
|
|
/* Hit end of trace data. */
|
|
return false;
|
|
}
|
|
|
|
/**
|
|
* fw_trace_get_first() - Find first valid entry in trace
|
|
* @trace_seq_data: Trace sequence data.
|
|
*
|
|
* Skips over invalid (usually zero) and ROGUE_FW_SF_FIRST entries.
|
|
*
|
|
* If the trace has no valid entries, this function will exit with the trace
|
|
* index pointing to the end of the trace. trace_seq_show() will return an error
|
|
* in this state.
|
|
*/
|
|
static void fw_trace_get_first(struct pvr_fw_trace_seq_data *trace_seq_data)
|
|
{
|
|
trace_seq_data->idx = 0;
|
|
|
|
while (trace_seq_data->idx < ROGUE_FW_TRACE_BUF_DEFAULT_SIZE_IN_DWORDS) {
|
|
u32 id = read_fw_trace(trace_seq_data, 0);
|
|
|
|
if (ROGUE_FW_LOG_VALIDID(id)) {
|
|
u32 sf_id = find_sfid(id);
|
|
|
|
if (sf_id != ROGUE_FW_SF_FIRST)
|
|
break;
|
|
}
|
|
trace_seq_data->idx++;
|
|
}
|
|
}
|
|
|
|
static void *fw_trace_seq_start(struct seq_file *s, loff_t *pos)
|
|
{
|
|
struct pvr_fw_trace_seq_data *trace_seq_data = s->private;
|
|
u32 i;
|
|
|
|
/* Reset trace index, then advance to *pos. */
|
|
fw_trace_get_first(trace_seq_data);
|
|
|
|
for (i = 0; i < *pos; i++) {
|
|
if (!fw_trace_get_next(trace_seq_data))
|
|
return NULL;
|
|
}
|
|
|
|
return (trace_seq_data->idx < ROGUE_FW_TRACE_BUF_DEFAULT_SIZE_IN_DWORDS) ? pos : NULL;
|
|
}
|
|
|
|
static void *fw_trace_seq_next(struct seq_file *s, void *v, loff_t *pos)
|
|
{
|
|
struct pvr_fw_trace_seq_data *trace_seq_data = s->private;
|
|
|
|
(*pos)++;
|
|
if (!fw_trace_get_next(trace_seq_data))
|
|
return NULL;
|
|
|
|
return (trace_seq_data->idx < ROGUE_FW_TRACE_BUF_DEFAULT_SIZE_IN_DWORDS) ? pos : NULL;
|
|
}
|
|
|
|
static void fw_trace_seq_stop(struct seq_file *s, void *v)
|
|
{
|
|
}
|
|
|
|
static int fw_trace_seq_show(struct seq_file *s, void *v)
|
|
{
|
|
struct pvr_fw_trace_seq_data *trace_seq_data = s->private;
|
|
u64 timestamp;
|
|
u32 id;
|
|
u32 sf_id;
|
|
|
|
if (trace_seq_data->idx >= ROGUE_FW_TRACE_BUF_DEFAULT_SIZE_IN_DWORDS)
|
|
return -EINVAL;
|
|
|
|
id = read_fw_trace(trace_seq_data, 0);
|
|
/* Index is not pointing at a valid entry. */
|
|
if (!ROGUE_FW_LOG_VALIDID(id))
|
|
return -EINVAL;
|
|
|
|
sf_id = find_sfid(id);
|
|
/* Index is not pointing at a valid entry. */
|
|
if (sf_id == ROGUE_FW_SF_LAST)
|
|
return -EINVAL;
|
|
|
|
timestamp = read_fw_trace(trace_seq_data, 1) |
|
|
((u64)read_fw_trace(trace_seq_data, 2) << 32);
|
|
timestamp = (timestamp & ~ROGUE_FWT_TIMESTAMP_TIME_CLRMSK) >>
|
|
ROGUE_FWT_TIMESTAMP_TIME_SHIFT;
|
|
|
|
seq_printf(s, "[%llu] : ", timestamp);
|
|
if (id == ROGUE_FW_SF_MAIN_ASSERT_FAILED) {
|
|
seq_printf(s, "ASSERTION %s failed at %s:%u",
|
|
trace_seq_data->assert_buf.info,
|
|
trace_seq_data->assert_buf.path,
|
|
trace_seq_data->assert_buf.line_num);
|
|
} else {
|
|
seq_printf(s, stid_fmts[sf_id].name,
|
|
read_fw_trace(trace_seq_data, 3),
|
|
read_fw_trace(trace_seq_data, 4),
|
|
read_fw_trace(trace_seq_data, 5),
|
|
read_fw_trace(trace_seq_data, 6),
|
|
read_fw_trace(trace_seq_data, 7),
|
|
read_fw_trace(trace_seq_data, 8),
|
|
read_fw_trace(trace_seq_data, 9),
|
|
read_fw_trace(trace_seq_data, 10),
|
|
read_fw_trace(trace_seq_data, 11),
|
|
read_fw_trace(trace_seq_data, 12),
|
|
read_fw_trace(trace_seq_data, 13),
|
|
read_fw_trace(trace_seq_data, 14),
|
|
read_fw_trace(trace_seq_data, 15),
|
|
read_fw_trace(trace_seq_data, 16),
|
|
read_fw_trace(trace_seq_data, 17),
|
|
read_fw_trace(trace_seq_data, 18),
|
|
read_fw_trace(trace_seq_data, 19),
|
|
read_fw_trace(trace_seq_data, 20),
|
|
read_fw_trace(trace_seq_data, 21),
|
|
read_fw_trace(trace_seq_data, 22));
|
|
}
|
|
seq_puts(s, "\n");
|
|
return 0;
|
|
}
|
|
|
|
static const struct seq_operations pvr_fw_trace_seq_ops = {
|
|
.start = fw_trace_seq_start,
|
|
.next = fw_trace_seq_next,
|
|
.stop = fw_trace_seq_stop,
|
|
.show = fw_trace_seq_show
|
|
};
|
|
|
|
static int fw_trace_open(struct inode *inode, struct file *file)
|
|
{
|
|
struct pvr_fw_trace_buffer *trace_buffer = inode->i_private;
|
|
struct rogue_fwif_tracebuf_space *tracebuf_space =
|
|
trace_buffer->tracebuf_space;
|
|
struct pvr_fw_trace_seq_data *trace_seq_data;
|
|
int err;
|
|
|
|
trace_seq_data = kzalloc(sizeof(*trace_seq_data), GFP_KERNEL);
|
|
if (!trace_seq_data)
|
|
return -ENOMEM;
|
|
|
|
trace_seq_data->buffer = kcalloc(ROGUE_FW_TRACE_BUF_DEFAULT_SIZE_IN_DWORDS,
|
|
sizeof(*trace_seq_data->buffer), GFP_KERNEL);
|
|
if (!trace_seq_data->buffer) {
|
|
err = -ENOMEM;
|
|
goto err_free_data;
|
|
}
|
|
|
|
/*
|
|
* Take a local copy of the trace buffer, as firmware may still be
|
|
* writing to it. This will exist as long as this file is open.
|
|
*/
|
|
memcpy(trace_seq_data->buffer, trace_buffer->buf,
|
|
ROGUE_FW_TRACE_BUF_DEFAULT_SIZE_IN_DWORDS * sizeof(u32));
|
|
trace_seq_data->start_offset = READ_ONCE(tracebuf_space->trace_pointer);
|
|
trace_seq_data->assert_buf = tracebuf_space->assert_buf;
|
|
fw_trace_get_first(trace_seq_data);
|
|
|
|
err = seq_open(file, &pvr_fw_trace_seq_ops);
|
|
if (err)
|
|
goto err_free_buffer;
|
|
|
|
((struct seq_file *)file->private_data)->private = trace_seq_data;
|
|
|
|
return 0;
|
|
|
|
err_free_buffer:
|
|
kfree(trace_seq_data->buffer);
|
|
|
|
err_free_data:
|
|
kfree(trace_seq_data);
|
|
|
|
return err;
|
|
}
|
|
|
|
static int fw_trace_release(struct inode *inode, struct file *file)
|
|
{
|
|
struct pvr_fw_trace_seq_data *trace_seq_data =
|
|
((struct seq_file *)file->private_data)->private;
|
|
|
|
seq_release(inode, file);
|
|
kfree(trace_seq_data->buffer);
|
|
kfree(trace_seq_data);
|
|
|
|
return 0;
|
|
}
|
|
|
|
static const struct file_operations pvr_fw_trace_fops = {
|
|
.owner = THIS_MODULE,
|
|
.open = fw_trace_open,
|
|
.read = seq_read,
|
|
.llseek = seq_lseek,
|
|
.release = fw_trace_release,
|
|
};
|
|
|
|
void
|
|
pvr_fw_trace_mask_update(struct pvr_device *pvr_dev, u32 old_mask, u32 new_mask)
|
|
{
|
|
if (old_mask != new_mask)
|
|
update_logtype(pvr_dev, new_mask);
|
|
}
|
|
|
|
void
|
|
pvr_fw_trace_debugfs_init(struct pvr_device *pvr_dev, struct dentry *dir)
|
|
{
|
|
struct pvr_fw_trace *fw_trace = &pvr_dev->fw_dev.fw_trace;
|
|
u32 thread_nr;
|
|
|
|
static_assert(ARRAY_SIZE(fw_trace->buffers) <= 10,
|
|
"The filename buffer is only large enough for a single-digit thread count");
|
|
|
|
for (thread_nr = 0; thread_nr < ARRAY_SIZE(fw_trace->buffers); ++thread_nr) {
|
|
char filename[8];
|
|
|
|
snprintf(filename, ARRAY_SIZE(filename), "trace_%u", thread_nr);
|
|
debugfs_create_file(filename, 0400, dir,
|
|
&fw_trace->buffers[thread_nr],
|
|
&pvr_fw_trace_fops);
|
|
}
|
|
}
|
|
#endif
|