Skip to content

Commit

Permalink
drm/i915/guc: Enable guc logging on guc log relay write
Browse files Browse the repository at this point in the history
Creating and opening the GuC log relay file enables and starts
the relay potentially before the caller is ready to consume logs.
Change the behavior so that relay starts only on an explicit call
to the write function (with a value of '1'). Other values flush
the log relay as before.

v2: Style changes and fix typos. Add guc_log_relay_stop()
function. (Daniele)

Cc: Matthew Brost <matthew.brost@intel.com>
Cc: Daniele Ceraolo Spurio <daniele.ceraolospurio@intel.com>
Cc: Michal Wajdeczko <michal.wajdeczko@intel.com>
Signed-off-by: Robert M. Fosha <robert.m.fosha@intel.com>
Reviewed-by: Matthew Brost <matthew.brost@intel.com>
Signed-off-by: Daniele Ceraolo Spurio <daniele.ceraolospurio@intel.com>
Link: https://patchwork.freedesktop.org/patch/msgid/20191022163754.23870-1-robert.m.fosha@intel.com
  • Loading branch information
Robert M. Fosha authored and Daniele Ceraolo Spurio committed Oct 23, 2019
1 parent 37c92dc commit 853ddb6
Show file tree
Hide file tree
Showing 3 changed files with 62 additions and 17 deletions.
53 changes: 41 additions & 12 deletions drivers/gpu/drm/i915/gt/uc/intel_guc_log.c
Original file line number Diff line number Diff line change
Expand Up @@ -226,7 +226,7 @@ static void guc_read_update_log_buffer(struct intel_guc_log *log)

mutex_lock(&log->relay.lock);

if (WARN_ON(!intel_guc_log_relay_enabled(log)))
if (WARN_ON(!intel_guc_log_relay_created(log)))
goto out_unlock;

/* Get the pointer to shared GuC log buffer */
Expand Down Expand Up @@ -361,6 +361,7 @@ void intel_guc_log_init_early(struct intel_guc_log *log)
{
mutex_init(&log->relay.lock);
INIT_WORK(&log->relay.flush_work, capture_logs_work);
log->relay.started = false;
}

static int guc_log_relay_create(struct intel_guc_log *log)
Expand Down Expand Up @@ -546,7 +547,7 @@ int intel_guc_log_set_level(struct intel_guc_log *log, u32 level)
return ret;
}

bool intel_guc_log_relay_enabled(const struct intel_guc_log *log)
bool intel_guc_log_relay_created(const struct intel_guc_log *log)
{
return log->relay.buf_addr;
}
Expand All @@ -560,7 +561,7 @@ int intel_guc_log_relay_open(struct intel_guc_log *log)

mutex_lock(&log->relay.lock);

if (intel_guc_log_relay_enabled(log)) {
if (intel_guc_log_relay_created(log)) {
ret = -EEXIST;
goto out_unlock;
}
Expand All @@ -585,6 +586,21 @@ int intel_guc_log_relay_open(struct intel_guc_log *log)

mutex_unlock(&log->relay.lock);

return 0;

out_relay:
guc_log_relay_destroy(log);
out_unlock:
mutex_unlock(&log->relay.lock);

return ret;
}

int intel_guc_log_relay_start(struct intel_guc_log *log)
{
if (log->relay.started)
return -EEXIST;

guc_log_enable_flush_events(log);

/*
Expand All @@ -594,21 +610,19 @@ int intel_guc_log_relay_open(struct intel_guc_log *log)
*/
queue_work(system_highpri_wq, &log->relay.flush_work);

return 0;

out_relay:
guc_log_relay_destroy(log);
out_unlock:
mutex_unlock(&log->relay.lock);
log->relay.started = true;

return ret;
return 0;
}

void intel_guc_log_relay_flush(struct intel_guc_log *log)
{
struct intel_guc *guc = log_to_guc(log);
intel_wakeref_t wakeref;

if (!log->relay.started)
return;

/*
* Before initiating the forceful flush, wait for any pending/ongoing
* flush to complete otherwise forceful flush may not actually happen.
Expand All @@ -622,18 +636,33 @@ void intel_guc_log_relay_flush(struct intel_guc_log *log)
guc_log_capture_logs(log);
}

void intel_guc_log_relay_close(struct intel_guc_log *log)
/*
* Stops the relay log. Called from intel_guc_log_relay_close(), so no
* possibility of race with start/flush since relay_write cannot race
* relay_close.
*/
static void guc_log_relay_stop(struct intel_guc_log *log)
{
struct intel_guc *guc = log_to_guc(log);
struct drm_i915_private *i915 = guc_to_gt(guc)->i915;

if (!log->relay.started)
return;

guc_log_disable_flush_events(log);
intel_synchronize_irq(i915);

flush_work(&log->relay.flush_work);

log->relay.started = false;
}

void intel_guc_log_relay_close(struct intel_guc_log *log)
{
guc_log_relay_stop(log);

mutex_lock(&log->relay.lock);
GEM_BUG_ON(!intel_guc_log_relay_enabled(log));
GEM_BUG_ON(!intel_guc_log_relay_created(log));
guc_log_unmap(log);
guc_log_relay_destroy(log);
mutex_unlock(&log->relay.lock);
Expand Down
4 changes: 3 additions & 1 deletion drivers/gpu/drm/i915/gt/uc/intel_guc_log.h
Original file line number Diff line number Diff line change
Expand Up @@ -47,6 +47,7 @@ struct intel_guc_log {
struct i915_vma *vma;
struct {
void *buf_addr;
bool started;
struct work_struct flush_work;
struct rchan *channel;
struct mutex lock;
Expand All @@ -65,8 +66,9 @@ int intel_guc_log_create(struct intel_guc_log *log);
void intel_guc_log_destroy(struct intel_guc_log *log);

int intel_guc_log_set_level(struct intel_guc_log *log, u32 level);
bool intel_guc_log_relay_enabled(const struct intel_guc_log *log);
bool intel_guc_log_relay_created(const struct intel_guc_log *log);
int intel_guc_log_relay_open(struct intel_guc_log *log);
int intel_guc_log_relay_start(struct intel_guc_log *log);
void intel_guc_log_relay_flush(struct intel_guc_log *log);
void intel_guc_log_relay_close(struct intel_guc_log *log);

Expand Down
22 changes: 18 additions & 4 deletions drivers/gpu/drm/i915/i915_debugfs.c
Original file line number Diff line number Diff line change
Expand Up @@ -1866,8 +1866,8 @@ static void i915_guc_log_info(struct seq_file *m,
struct intel_guc_log *log = &dev_priv->gt.uc.guc.log;
enum guc_log_buffer_type type;

if (!intel_guc_log_relay_enabled(log)) {
seq_puts(m, "GuC log relay disabled\n");
if (!intel_guc_log_relay_created(log)) {
seq_puts(m, "GuC log relay not created\n");
return;
}

Expand Down Expand Up @@ -2054,9 +2054,23 @@ i915_guc_log_relay_write(struct file *filp,
loff_t *ppos)
{
struct intel_guc_log *log = filp->private_data;
int val;
int ret;

intel_guc_log_relay_flush(log);
return cnt;
ret = kstrtoint_from_user(ubuf, cnt, 0, &val);
if (ret < 0)
return ret;

/*
* Enable and start the guc log relay on value of 1.
* Flush log relay for any other value.
*/
if (val == 1)
ret = intel_guc_log_relay_start(log);
else
intel_guc_log_relay_flush(log);

return ret ?: cnt;
}

static int i915_guc_log_relay_release(struct inode *inode, struct file *file)
Expand Down

0 comments on commit 853ddb6

Please sign in to comment.