03a2a60606
Commit750e76b4f9
("drm/i915/gt: Move the [class][inst] lookup for engines onto the GT") changed the engine query to iterate over uabi engines but left the buffer size calculation look at the physical engine count. Difference has no practical consequence but it is nicer to align both queries. Signed-off-by: Tvrtko Ursulin <tvrtko.ursulin@intel.com> Fixes:750e76b4f9
("drm/i915/gt: Move the [class][inst] lookup for engines onto the GT") Cc: Chris Wilson <chris@chris-wilson.co.uk> Cc: Daniele Ceraolo Spurio <daniele.ceraolospurio@intel.com> Reviewed-by: Chris Wilson <chris@chris-wilson.co.uk> Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk> Link: https://patchwork.freedesktop.org/patch/msgid/20191122104115.29610-1-tvrtko.ursulin@linux.intel.com (cherry picked from commit9acc99d8f2
) Signed-off-by: Joonas Lahtinen <joonas.lahtinen@linux.intel.com>
489 lines
12 KiB
C
489 lines
12 KiB
C
/*
|
|
* SPDX-License-Identifier: MIT
|
|
*
|
|
* Copyright © 2018 Intel Corporation
|
|
*/
|
|
|
|
#include <linux/nospec.h>
|
|
|
|
#include "i915_drv.h"
|
|
#include "i915_perf.h"
|
|
#include "i915_query.h"
|
|
#include <uapi/drm/i915_drm.h>
|
|
|
|
static int copy_query_item(void *query_hdr, size_t query_sz,
|
|
u32 total_length,
|
|
struct drm_i915_query_item *query_item)
|
|
{
|
|
if (query_item->length == 0)
|
|
return total_length;
|
|
|
|
if (query_item->length < total_length)
|
|
return -EINVAL;
|
|
|
|
if (copy_from_user(query_hdr, u64_to_user_ptr(query_item->data_ptr),
|
|
query_sz))
|
|
return -EFAULT;
|
|
|
|
if (!access_ok(u64_to_user_ptr(query_item->data_ptr),
|
|
total_length))
|
|
return -EFAULT;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int query_topology_info(struct drm_i915_private *dev_priv,
|
|
struct drm_i915_query_item *query_item)
|
|
{
|
|
const struct sseu_dev_info *sseu = &RUNTIME_INFO(dev_priv)->sseu;
|
|
struct drm_i915_query_topology_info topo;
|
|
u32 slice_length, subslice_length, eu_length, total_length;
|
|
int ret;
|
|
|
|
if (query_item->flags != 0)
|
|
return -EINVAL;
|
|
|
|
if (sseu->max_slices == 0)
|
|
return -ENODEV;
|
|
|
|
BUILD_BUG_ON(sizeof(u8) != sizeof(sseu->slice_mask));
|
|
|
|
slice_length = sizeof(sseu->slice_mask);
|
|
subslice_length = sseu->max_slices * sseu->ss_stride;
|
|
eu_length = sseu->max_slices * sseu->max_subslices * sseu->eu_stride;
|
|
total_length = sizeof(topo) + slice_length + subslice_length +
|
|
eu_length;
|
|
|
|
ret = copy_query_item(&topo, sizeof(topo), total_length,
|
|
query_item);
|
|
if (ret != 0)
|
|
return ret;
|
|
|
|
if (topo.flags != 0)
|
|
return -EINVAL;
|
|
|
|
memset(&topo, 0, sizeof(topo));
|
|
topo.max_slices = sseu->max_slices;
|
|
topo.max_subslices = sseu->max_subslices;
|
|
topo.max_eus_per_subslice = sseu->max_eus_per_subslice;
|
|
|
|
topo.subslice_offset = slice_length;
|
|
topo.subslice_stride = sseu->ss_stride;
|
|
topo.eu_offset = slice_length + subslice_length;
|
|
topo.eu_stride = sseu->eu_stride;
|
|
|
|
if (__copy_to_user(u64_to_user_ptr(query_item->data_ptr),
|
|
&topo, sizeof(topo)))
|
|
return -EFAULT;
|
|
|
|
if (__copy_to_user(u64_to_user_ptr(query_item->data_ptr + sizeof(topo)),
|
|
&sseu->slice_mask, slice_length))
|
|
return -EFAULT;
|
|
|
|
if (__copy_to_user(u64_to_user_ptr(query_item->data_ptr +
|
|
sizeof(topo) + slice_length),
|
|
sseu->subslice_mask, subslice_length))
|
|
return -EFAULT;
|
|
|
|
if (__copy_to_user(u64_to_user_ptr(query_item->data_ptr +
|
|
sizeof(topo) +
|
|
slice_length + subslice_length),
|
|
sseu->eu_mask, eu_length))
|
|
return -EFAULT;
|
|
|
|
return total_length;
|
|
}
|
|
|
|
static int
|
|
query_engine_info(struct drm_i915_private *i915,
|
|
struct drm_i915_query_item *query_item)
|
|
{
|
|
struct drm_i915_query_engine_info __user *query_ptr =
|
|
u64_to_user_ptr(query_item->data_ptr);
|
|
struct drm_i915_engine_info __user *info_ptr;
|
|
struct drm_i915_query_engine_info query;
|
|
struct drm_i915_engine_info info = { };
|
|
unsigned int num_uabi_engines = 0;
|
|
struct intel_engine_cs *engine;
|
|
int len, ret;
|
|
|
|
if (query_item->flags)
|
|
return -EINVAL;
|
|
|
|
for_each_uabi_engine(engine, i915)
|
|
num_uabi_engines++;
|
|
|
|
len = sizeof(struct drm_i915_query_engine_info) +
|
|
num_uabi_engines * sizeof(struct drm_i915_engine_info);
|
|
|
|
ret = copy_query_item(&query, sizeof(query), len, query_item);
|
|
if (ret != 0)
|
|
return ret;
|
|
|
|
if (query.num_engines || query.rsvd[0] || query.rsvd[1] ||
|
|
query.rsvd[2])
|
|
return -EINVAL;
|
|
|
|
info_ptr = &query_ptr->engines[0];
|
|
|
|
for_each_uabi_engine(engine, i915) {
|
|
info.engine.engine_class = engine->uabi_class;
|
|
info.engine.engine_instance = engine->uabi_instance;
|
|
info.capabilities = engine->uabi_capabilities;
|
|
|
|
if (__copy_to_user(info_ptr, &info, sizeof(info)))
|
|
return -EFAULT;
|
|
|
|
query.num_engines++;
|
|
info_ptr++;
|
|
}
|
|
|
|
if (__copy_to_user(query_ptr, &query, sizeof(query)))
|
|
return -EFAULT;
|
|
|
|
return len;
|
|
}
|
|
|
|
static int can_copy_perf_config_registers_or_number(u32 user_n_regs,
|
|
u64 user_regs_ptr,
|
|
u32 kernel_n_regs)
|
|
{
|
|
/*
|
|
* We'll just put the number of registers, and won't copy the
|
|
* register.
|
|
*/
|
|
if (user_n_regs == 0)
|
|
return 0;
|
|
|
|
if (user_n_regs < kernel_n_regs)
|
|
return -EINVAL;
|
|
|
|
if (!access_ok(u64_to_user_ptr(user_regs_ptr),
|
|
2 * sizeof(u32) * kernel_n_regs))
|
|
return -EFAULT;
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int copy_perf_config_registers_or_number(const struct i915_oa_reg *kernel_regs,
|
|
u32 kernel_n_regs,
|
|
u64 user_regs_ptr,
|
|
u32 *user_n_regs)
|
|
{
|
|
u32 r;
|
|
|
|
if (*user_n_regs == 0) {
|
|
*user_n_regs = kernel_n_regs;
|
|
return 0;
|
|
}
|
|
|
|
*user_n_regs = kernel_n_regs;
|
|
|
|
for (r = 0; r < kernel_n_regs; r++) {
|
|
u32 __user *user_reg_ptr =
|
|
u64_to_user_ptr(user_regs_ptr + sizeof(u32) * r * 2);
|
|
u32 __user *user_val_ptr =
|
|
u64_to_user_ptr(user_regs_ptr + sizeof(u32) * r * 2 +
|
|
sizeof(u32));
|
|
int ret;
|
|
|
|
ret = __put_user(i915_mmio_reg_offset(kernel_regs[r].addr),
|
|
user_reg_ptr);
|
|
if (ret)
|
|
return -EFAULT;
|
|
|
|
ret = __put_user(kernel_regs[r].value, user_val_ptr);
|
|
if (ret)
|
|
return -EFAULT;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
static int query_perf_config_data(struct drm_i915_private *i915,
|
|
struct drm_i915_query_item *query_item,
|
|
bool use_uuid)
|
|
{
|
|
struct drm_i915_query_perf_config __user *user_query_config_ptr =
|
|
u64_to_user_ptr(query_item->data_ptr);
|
|
struct drm_i915_perf_oa_config __user *user_config_ptr =
|
|
u64_to_user_ptr(query_item->data_ptr +
|
|
sizeof(struct drm_i915_query_perf_config));
|
|
struct drm_i915_perf_oa_config user_config;
|
|
struct i915_perf *perf = &i915->perf;
|
|
struct i915_oa_config *oa_config;
|
|
char uuid[UUID_STRING_LEN + 1];
|
|
u64 config_id;
|
|
u32 flags, total_size;
|
|
int ret;
|
|
|
|
if (!perf->i915)
|
|
return -ENODEV;
|
|
|
|
total_size =
|
|
sizeof(struct drm_i915_query_perf_config) +
|
|
sizeof(struct drm_i915_perf_oa_config);
|
|
|
|
if (query_item->length == 0)
|
|
return total_size;
|
|
|
|
if (query_item->length < total_size) {
|
|
DRM_DEBUG("Invalid query config data item size=%u expected=%u\n",
|
|
query_item->length, total_size);
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (!access_ok(user_query_config_ptr, total_size))
|
|
return -EFAULT;
|
|
|
|
if (__get_user(flags, &user_query_config_ptr->flags))
|
|
return -EFAULT;
|
|
|
|
if (flags != 0)
|
|
return -EINVAL;
|
|
|
|
if (use_uuid) {
|
|
struct i915_oa_config *tmp;
|
|
int id;
|
|
|
|
BUILD_BUG_ON(sizeof(user_query_config_ptr->uuid) >= sizeof(uuid));
|
|
|
|
memset(&uuid, 0, sizeof(uuid));
|
|
if (__copy_from_user(uuid, user_query_config_ptr->uuid,
|
|
sizeof(user_query_config_ptr->uuid)))
|
|
return -EFAULT;
|
|
|
|
oa_config = NULL;
|
|
rcu_read_lock();
|
|
idr_for_each_entry(&perf->metrics_idr, tmp, id) {
|
|
if (!strcmp(tmp->uuid, uuid)) {
|
|
oa_config = i915_oa_config_get(tmp);
|
|
break;
|
|
}
|
|
}
|
|
rcu_read_unlock();
|
|
} else {
|
|
if (__get_user(config_id, &user_query_config_ptr->config))
|
|
return -EFAULT;
|
|
|
|
oa_config = i915_perf_get_oa_config(perf, config_id);
|
|
}
|
|
if (!oa_config)
|
|
return -ENOENT;
|
|
|
|
if (__copy_from_user(&user_config, user_config_ptr,
|
|
sizeof(user_config))) {
|
|
ret = -EFAULT;
|
|
goto out;
|
|
}
|
|
|
|
ret = can_copy_perf_config_registers_or_number(user_config.n_boolean_regs,
|
|
user_config.boolean_regs_ptr,
|
|
oa_config->b_counter_regs_len);
|
|
if (ret)
|
|
goto out;
|
|
|
|
ret = can_copy_perf_config_registers_or_number(user_config.n_flex_regs,
|
|
user_config.flex_regs_ptr,
|
|
oa_config->flex_regs_len);
|
|
if (ret)
|
|
goto out;
|
|
|
|
ret = can_copy_perf_config_registers_or_number(user_config.n_mux_regs,
|
|
user_config.mux_regs_ptr,
|
|
oa_config->mux_regs_len);
|
|
if (ret)
|
|
goto out;
|
|
|
|
ret = copy_perf_config_registers_or_number(oa_config->b_counter_regs,
|
|
oa_config->b_counter_regs_len,
|
|
user_config.boolean_regs_ptr,
|
|
&user_config.n_boolean_regs);
|
|
if (ret)
|
|
goto out;
|
|
|
|
ret = copy_perf_config_registers_or_number(oa_config->flex_regs,
|
|
oa_config->flex_regs_len,
|
|
user_config.flex_regs_ptr,
|
|
&user_config.n_flex_regs);
|
|
if (ret)
|
|
goto out;
|
|
|
|
ret = copy_perf_config_registers_or_number(oa_config->mux_regs,
|
|
oa_config->mux_regs_len,
|
|
user_config.mux_regs_ptr,
|
|
&user_config.n_mux_regs);
|
|
if (ret)
|
|
goto out;
|
|
|
|
memcpy(user_config.uuid, oa_config->uuid, sizeof(user_config.uuid));
|
|
|
|
if (__copy_to_user(user_config_ptr, &user_config,
|
|
sizeof(user_config))) {
|
|
ret = -EFAULT;
|
|
goto out;
|
|
}
|
|
|
|
ret = total_size;
|
|
|
|
out:
|
|
i915_oa_config_put(oa_config);
|
|
return ret;
|
|
}
|
|
|
|
static size_t sizeof_perf_config_list(size_t count)
|
|
{
|
|
return sizeof(struct drm_i915_query_perf_config) + sizeof(u64) * count;
|
|
}
|
|
|
|
static size_t sizeof_perf_metrics(struct i915_perf *perf)
|
|
{
|
|
struct i915_oa_config *tmp;
|
|
size_t i;
|
|
int id;
|
|
|
|
i = 1;
|
|
rcu_read_lock();
|
|
idr_for_each_entry(&perf->metrics_idr, tmp, id)
|
|
i++;
|
|
rcu_read_unlock();
|
|
|
|
return sizeof_perf_config_list(i);
|
|
}
|
|
|
|
static int query_perf_config_list(struct drm_i915_private *i915,
|
|
struct drm_i915_query_item *query_item)
|
|
{
|
|
struct drm_i915_query_perf_config __user *user_query_config_ptr =
|
|
u64_to_user_ptr(query_item->data_ptr);
|
|
struct i915_perf *perf = &i915->perf;
|
|
u64 *oa_config_ids = NULL;
|
|
int alloc, n_configs;
|
|
u32 flags;
|
|
int ret;
|
|
|
|
if (!perf->i915)
|
|
return -ENODEV;
|
|
|
|
if (query_item->length == 0)
|
|
return sizeof_perf_metrics(perf);
|
|
|
|
if (get_user(flags, &user_query_config_ptr->flags))
|
|
return -EFAULT;
|
|
|
|
if (flags != 0)
|
|
return -EINVAL;
|
|
|
|
n_configs = 1;
|
|
do {
|
|
struct i915_oa_config *tmp;
|
|
u64 *ids;
|
|
int id;
|
|
|
|
ids = krealloc(oa_config_ids,
|
|
n_configs * sizeof(*oa_config_ids),
|
|
GFP_KERNEL);
|
|
if (!ids)
|
|
return -ENOMEM;
|
|
|
|
alloc = fetch_and_zero(&n_configs);
|
|
|
|
ids[n_configs++] = 1ull; /* reserved for test_config */
|
|
rcu_read_lock();
|
|
idr_for_each_entry(&perf->metrics_idr, tmp, id) {
|
|
if (n_configs < alloc)
|
|
ids[n_configs] = id;
|
|
n_configs++;
|
|
}
|
|
rcu_read_unlock();
|
|
|
|
oa_config_ids = ids;
|
|
} while (n_configs > alloc);
|
|
|
|
if (query_item->length < sizeof_perf_config_list(n_configs)) {
|
|
DRM_DEBUG("Invalid query config list item size=%u expected=%zu\n",
|
|
query_item->length,
|
|
sizeof_perf_config_list(n_configs));
|
|
kfree(oa_config_ids);
|
|
return -EINVAL;
|
|
}
|
|
|
|
if (put_user(n_configs, &user_query_config_ptr->config)) {
|
|
kfree(oa_config_ids);
|
|
return -EFAULT;
|
|
}
|
|
|
|
ret = copy_to_user(user_query_config_ptr + 1,
|
|
oa_config_ids,
|
|
n_configs * sizeof(*oa_config_ids));
|
|
kfree(oa_config_ids);
|
|
if (ret)
|
|
return -EFAULT;
|
|
|
|
return sizeof_perf_config_list(n_configs);
|
|
}
|
|
|
|
static int query_perf_config(struct drm_i915_private *i915,
|
|
struct drm_i915_query_item *query_item)
|
|
{
|
|
switch (query_item->flags) {
|
|
case DRM_I915_QUERY_PERF_CONFIG_LIST:
|
|
return query_perf_config_list(i915, query_item);
|
|
case DRM_I915_QUERY_PERF_CONFIG_DATA_FOR_UUID:
|
|
return query_perf_config_data(i915, query_item, true);
|
|
case DRM_I915_QUERY_PERF_CONFIG_DATA_FOR_ID:
|
|
return query_perf_config_data(i915, query_item, false);
|
|
default:
|
|
return -EINVAL;
|
|
}
|
|
}
|
|
|
|
static int (* const i915_query_funcs[])(struct drm_i915_private *dev_priv,
|
|
struct drm_i915_query_item *query_item) = {
|
|
query_topology_info,
|
|
query_engine_info,
|
|
query_perf_config,
|
|
};
|
|
|
|
int i915_query_ioctl(struct drm_device *dev, void *data, struct drm_file *file)
|
|
{
|
|
struct drm_i915_private *dev_priv = to_i915(dev);
|
|
struct drm_i915_query *args = data;
|
|
struct drm_i915_query_item __user *user_item_ptr =
|
|
u64_to_user_ptr(args->items_ptr);
|
|
u32 i;
|
|
|
|
if (args->flags != 0)
|
|
return -EINVAL;
|
|
|
|
for (i = 0; i < args->num_items; i++, user_item_ptr++) {
|
|
struct drm_i915_query_item item;
|
|
unsigned long func_idx;
|
|
int ret;
|
|
|
|
if (copy_from_user(&item, user_item_ptr, sizeof(item)))
|
|
return -EFAULT;
|
|
|
|
if (item.query_id == 0)
|
|
return -EINVAL;
|
|
|
|
if (overflows_type(item.query_id - 1, unsigned long))
|
|
return -EINVAL;
|
|
|
|
func_idx = item.query_id - 1;
|
|
|
|
ret = -EINVAL;
|
|
if (func_idx < ARRAY_SIZE(i915_query_funcs)) {
|
|
func_idx = array_index_nospec(func_idx,
|
|
ARRAY_SIZE(i915_query_funcs));
|
|
ret = i915_query_funcs[func_idx](dev_priv, &item);
|
|
}
|
|
|
|
/* Only write the length back to userspace if they differ. */
|
|
if (ret != item.length && put_user(ret, &user_item_ptr->length))
|
|
return -EFAULT;
|
|
}
|
|
|
|
return 0;
|
|
}
|