162306a36Sopenharmony_ci/* 262306a36Sopenharmony_ci * SPDX-License-Identifier: MIT 362306a36Sopenharmony_ci * 462306a36Sopenharmony_ci * Copyright © 2018 Intel Corporation 562306a36Sopenharmony_ci */ 662306a36Sopenharmony_ci 762306a36Sopenharmony_ci#include <linux/nospec.h> 862306a36Sopenharmony_ci 962306a36Sopenharmony_ci#include "i915_drv.h" 1062306a36Sopenharmony_ci#include "i915_perf.h" 1162306a36Sopenharmony_ci#include "i915_query.h" 1262306a36Sopenharmony_ci#include "gt/intel_engine_user.h" 1362306a36Sopenharmony_ci#include <uapi/drm/i915_drm.h> 1462306a36Sopenharmony_ci 1562306a36Sopenharmony_cistatic int copy_query_item(void *query_hdr, size_t query_sz, 1662306a36Sopenharmony_ci u32 total_length, 1762306a36Sopenharmony_ci struct drm_i915_query_item *query_item) 1862306a36Sopenharmony_ci{ 1962306a36Sopenharmony_ci if (query_item->length == 0) 2062306a36Sopenharmony_ci return total_length; 2162306a36Sopenharmony_ci 2262306a36Sopenharmony_ci if (query_item->length < total_length) 2362306a36Sopenharmony_ci return -EINVAL; 2462306a36Sopenharmony_ci 2562306a36Sopenharmony_ci if (copy_from_user(query_hdr, u64_to_user_ptr(query_item->data_ptr), 2662306a36Sopenharmony_ci query_sz)) 2762306a36Sopenharmony_ci return -EFAULT; 2862306a36Sopenharmony_ci 2962306a36Sopenharmony_ci return 0; 3062306a36Sopenharmony_ci} 3162306a36Sopenharmony_ci 3262306a36Sopenharmony_cistatic int fill_topology_info(const struct sseu_dev_info *sseu, 3362306a36Sopenharmony_ci struct drm_i915_query_item *query_item, 3462306a36Sopenharmony_ci intel_sseu_ss_mask_t subslice_mask) 3562306a36Sopenharmony_ci{ 3662306a36Sopenharmony_ci struct drm_i915_query_topology_info topo; 3762306a36Sopenharmony_ci u32 slice_length, subslice_length, eu_length, total_length; 3862306a36Sopenharmony_ci int ss_stride = GEN_SSEU_STRIDE(sseu->max_subslices); 3962306a36Sopenharmony_ci int eu_stride = GEN_SSEU_STRIDE(sseu->max_eus_per_subslice); 4062306a36Sopenharmony_ci int ret; 4162306a36Sopenharmony_ci 4262306a36Sopenharmony_ci BUILD_BUG_ON(sizeof(u8) != sizeof(sseu->slice_mask)); 4362306a36Sopenharmony_ci 4462306a36Sopenharmony_ci if (sseu->max_slices == 0) 4562306a36Sopenharmony_ci return -ENODEV; 4662306a36Sopenharmony_ci 4762306a36Sopenharmony_ci slice_length = sizeof(sseu->slice_mask); 4862306a36Sopenharmony_ci subslice_length = sseu->max_slices * ss_stride; 4962306a36Sopenharmony_ci eu_length = sseu->max_slices * sseu->max_subslices * eu_stride; 5062306a36Sopenharmony_ci total_length = sizeof(topo) + slice_length + subslice_length + 5162306a36Sopenharmony_ci eu_length; 5262306a36Sopenharmony_ci 5362306a36Sopenharmony_ci ret = copy_query_item(&topo, sizeof(topo), total_length, query_item); 5462306a36Sopenharmony_ci 5562306a36Sopenharmony_ci if (ret != 0) 5662306a36Sopenharmony_ci return ret; 5762306a36Sopenharmony_ci 5862306a36Sopenharmony_ci memset(&topo, 0, sizeof(topo)); 5962306a36Sopenharmony_ci topo.max_slices = sseu->max_slices; 6062306a36Sopenharmony_ci topo.max_subslices = sseu->max_subslices; 6162306a36Sopenharmony_ci topo.max_eus_per_subslice = sseu->max_eus_per_subslice; 6262306a36Sopenharmony_ci 6362306a36Sopenharmony_ci topo.subslice_offset = slice_length; 6462306a36Sopenharmony_ci topo.subslice_stride = ss_stride; 6562306a36Sopenharmony_ci topo.eu_offset = slice_length + subslice_length; 6662306a36Sopenharmony_ci topo.eu_stride = eu_stride; 6762306a36Sopenharmony_ci 6862306a36Sopenharmony_ci if (copy_to_user(u64_to_user_ptr(query_item->data_ptr), 6962306a36Sopenharmony_ci &topo, sizeof(topo))) 7062306a36Sopenharmony_ci return -EFAULT; 7162306a36Sopenharmony_ci 7262306a36Sopenharmony_ci if (copy_to_user(u64_to_user_ptr(query_item->data_ptr + sizeof(topo)), 7362306a36Sopenharmony_ci &sseu->slice_mask, slice_length)) 7462306a36Sopenharmony_ci return -EFAULT; 7562306a36Sopenharmony_ci 7662306a36Sopenharmony_ci if (intel_sseu_copy_ssmask_to_user(u64_to_user_ptr(query_item->data_ptr + 7762306a36Sopenharmony_ci sizeof(topo) + slice_length), 7862306a36Sopenharmony_ci sseu)) 7962306a36Sopenharmony_ci return -EFAULT; 8062306a36Sopenharmony_ci 8162306a36Sopenharmony_ci if (intel_sseu_copy_eumask_to_user(u64_to_user_ptr(query_item->data_ptr + 8262306a36Sopenharmony_ci sizeof(topo) + 8362306a36Sopenharmony_ci slice_length + subslice_length), 8462306a36Sopenharmony_ci sseu)) 8562306a36Sopenharmony_ci return -EFAULT; 8662306a36Sopenharmony_ci 8762306a36Sopenharmony_ci return total_length; 8862306a36Sopenharmony_ci} 8962306a36Sopenharmony_ci 9062306a36Sopenharmony_cistatic int query_topology_info(struct drm_i915_private *dev_priv, 9162306a36Sopenharmony_ci struct drm_i915_query_item *query_item) 9262306a36Sopenharmony_ci{ 9362306a36Sopenharmony_ci const struct sseu_dev_info *sseu = &to_gt(dev_priv)->info.sseu; 9462306a36Sopenharmony_ci 9562306a36Sopenharmony_ci if (query_item->flags != 0) 9662306a36Sopenharmony_ci return -EINVAL; 9762306a36Sopenharmony_ci 9862306a36Sopenharmony_ci return fill_topology_info(sseu, query_item, sseu->subslice_mask); 9962306a36Sopenharmony_ci} 10062306a36Sopenharmony_ci 10162306a36Sopenharmony_cistatic int query_geometry_subslices(struct drm_i915_private *i915, 10262306a36Sopenharmony_ci struct drm_i915_query_item *query_item) 10362306a36Sopenharmony_ci{ 10462306a36Sopenharmony_ci const struct sseu_dev_info *sseu; 10562306a36Sopenharmony_ci struct intel_engine_cs *engine; 10662306a36Sopenharmony_ci struct i915_engine_class_instance classinstance; 10762306a36Sopenharmony_ci 10862306a36Sopenharmony_ci if (GRAPHICS_VER_FULL(i915) < IP_VER(12, 50)) 10962306a36Sopenharmony_ci return -ENODEV; 11062306a36Sopenharmony_ci 11162306a36Sopenharmony_ci classinstance = *((struct i915_engine_class_instance *)&query_item->flags); 11262306a36Sopenharmony_ci 11362306a36Sopenharmony_ci engine = intel_engine_lookup_user(i915, (u8)classinstance.engine_class, 11462306a36Sopenharmony_ci (u8)classinstance.engine_instance); 11562306a36Sopenharmony_ci 11662306a36Sopenharmony_ci if (!engine) 11762306a36Sopenharmony_ci return -EINVAL; 11862306a36Sopenharmony_ci 11962306a36Sopenharmony_ci if (engine->class != RENDER_CLASS) 12062306a36Sopenharmony_ci return -EINVAL; 12162306a36Sopenharmony_ci 12262306a36Sopenharmony_ci sseu = &engine->gt->info.sseu; 12362306a36Sopenharmony_ci 12462306a36Sopenharmony_ci return fill_topology_info(sseu, query_item, sseu->geometry_subslice_mask); 12562306a36Sopenharmony_ci} 12662306a36Sopenharmony_ci 12762306a36Sopenharmony_cistatic int 12862306a36Sopenharmony_ciquery_engine_info(struct drm_i915_private *i915, 12962306a36Sopenharmony_ci struct drm_i915_query_item *query_item) 13062306a36Sopenharmony_ci{ 13162306a36Sopenharmony_ci struct drm_i915_query_engine_info __user *query_ptr = 13262306a36Sopenharmony_ci u64_to_user_ptr(query_item->data_ptr); 13362306a36Sopenharmony_ci struct drm_i915_engine_info __user *info_ptr; 13462306a36Sopenharmony_ci struct drm_i915_query_engine_info query; 13562306a36Sopenharmony_ci struct drm_i915_engine_info info = { }; 13662306a36Sopenharmony_ci unsigned int num_uabi_engines = 0; 13762306a36Sopenharmony_ci struct intel_engine_cs *engine; 13862306a36Sopenharmony_ci int len, ret; 13962306a36Sopenharmony_ci 14062306a36Sopenharmony_ci if (query_item->flags) 14162306a36Sopenharmony_ci return -EINVAL; 14262306a36Sopenharmony_ci 14362306a36Sopenharmony_ci for_each_uabi_engine(engine, i915) 14462306a36Sopenharmony_ci num_uabi_engines++; 14562306a36Sopenharmony_ci 14662306a36Sopenharmony_ci len = struct_size(query_ptr, engines, num_uabi_engines); 14762306a36Sopenharmony_ci 14862306a36Sopenharmony_ci ret = copy_query_item(&query, sizeof(query), len, query_item); 14962306a36Sopenharmony_ci if (ret != 0) 15062306a36Sopenharmony_ci return ret; 15162306a36Sopenharmony_ci 15262306a36Sopenharmony_ci if (query.num_engines || query.rsvd[0] || query.rsvd[1] || 15362306a36Sopenharmony_ci query.rsvd[2]) 15462306a36Sopenharmony_ci return -EINVAL; 15562306a36Sopenharmony_ci 15662306a36Sopenharmony_ci info_ptr = &query_ptr->engines[0]; 15762306a36Sopenharmony_ci 15862306a36Sopenharmony_ci for_each_uabi_engine(engine, i915) { 15962306a36Sopenharmony_ci info.engine.engine_class = engine->uabi_class; 16062306a36Sopenharmony_ci info.engine.engine_instance = engine->uabi_instance; 16162306a36Sopenharmony_ci info.flags = I915_ENGINE_INFO_HAS_LOGICAL_INSTANCE; 16262306a36Sopenharmony_ci info.capabilities = engine->uabi_capabilities; 16362306a36Sopenharmony_ci info.logical_instance = ilog2(engine->logical_mask); 16462306a36Sopenharmony_ci 16562306a36Sopenharmony_ci if (copy_to_user(info_ptr, &info, sizeof(info))) 16662306a36Sopenharmony_ci return -EFAULT; 16762306a36Sopenharmony_ci 16862306a36Sopenharmony_ci query.num_engines++; 16962306a36Sopenharmony_ci info_ptr++; 17062306a36Sopenharmony_ci } 17162306a36Sopenharmony_ci 17262306a36Sopenharmony_ci if (copy_to_user(query_ptr, &query, sizeof(query))) 17362306a36Sopenharmony_ci return -EFAULT; 17462306a36Sopenharmony_ci 17562306a36Sopenharmony_ci return len; 17662306a36Sopenharmony_ci} 17762306a36Sopenharmony_ci 17862306a36Sopenharmony_cistatic int can_copy_perf_config_registers_or_number(u32 user_n_regs, 17962306a36Sopenharmony_ci u64 user_regs_ptr, 18062306a36Sopenharmony_ci u32 kernel_n_regs) 18162306a36Sopenharmony_ci{ 18262306a36Sopenharmony_ci /* 18362306a36Sopenharmony_ci * We'll just put the number of registers, and won't copy the 18462306a36Sopenharmony_ci * register. 18562306a36Sopenharmony_ci */ 18662306a36Sopenharmony_ci if (user_n_regs == 0) 18762306a36Sopenharmony_ci return 0; 18862306a36Sopenharmony_ci 18962306a36Sopenharmony_ci if (user_n_regs < kernel_n_regs) 19062306a36Sopenharmony_ci return -EINVAL; 19162306a36Sopenharmony_ci 19262306a36Sopenharmony_ci return 0; 19362306a36Sopenharmony_ci} 19462306a36Sopenharmony_ci 19562306a36Sopenharmony_cistatic int copy_perf_config_registers_or_number(const struct i915_oa_reg *kernel_regs, 19662306a36Sopenharmony_ci u32 kernel_n_regs, 19762306a36Sopenharmony_ci u64 user_regs_ptr, 19862306a36Sopenharmony_ci u32 *user_n_regs) 19962306a36Sopenharmony_ci{ 20062306a36Sopenharmony_ci u32 __user *p = u64_to_user_ptr(user_regs_ptr); 20162306a36Sopenharmony_ci u32 r; 20262306a36Sopenharmony_ci 20362306a36Sopenharmony_ci if (*user_n_regs == 0) { 20462306a36Sopenharmony_ci *user_n_regs = kernel_n_regs; 20562306a36Sopenharmony_ci return 0; 20662306a36Sopenharmony_ci } 20762306a36Sopenharmony_ci 20862306a36Sopenharmony_ci *user_n_regs = kernel_n_regs; 20962306a36Sopenharmony_ci 21062306a36Sopenharmony_ci if (!user_write_access_begin(p, 2 * sizeof(u32) * kernel_n_regs)) 21162306a36Sopenharmony_ci return -EFAULT; 21262306a36Sopenharmony_ci 21362306a36Sopenharmony_ci for (r = 0; r < kernel_n_regs; r++, p += 2) { 21462306a36Sopenharmony_ci unsafe_put_user(i915_mmio_reg_offset(kernel_regs[r].addr), 21562306a36Sopenharmony_ci p, Efault); 21662306a36Sopenharmony_ci unsafe_put_user(kernel_regs[r].value, p + 1, Efault); 21762306a36Sopenharmony_ci } 21862306a36Sopenharmony_ci user_write_access_end(); 21962306a36Sopenharmony_ci return 0; 22062306a36Sopenharmony_ciEfault: 22162306a36Sopenharmony_ci user_write_access_end(); 22262306a36Sopenharmony_ci return -EFAULT; 22362306a36Sopenharmony_ci} 22462306a36Sopenharmony_ci 22562306a36Sopenharmony_cistatic int query_perf_config_data(struct drm_i915_private *i915, 22662306a36Sopenharmony_ci struct drm_i915_query_item *query_item, 22762306a36Sopenharmony_ci bool use_uuid) 22862306a36Sopenharmony_ci{ 22962306a36Sopenharmony_ci struct drm_i915_query_perf_config __user *user_query_config_ptr = 23062306a36Sopenharmony_ci u64_to_user_ptr(query_item->data_ptr); 23162306a36Sopenharmony_ci struct drm_i915_perf_oa_config __user *user_config_ptr = 23262306a36Sopenharmony_ci u64_to_user_ptr(query_item->data_ptr + 23362306a36Sopenharmony_ci sizeof(struct drm_i915_query_perf_config)); 23462306a36Sopenharmony_ci struct drm_i915_perf_oa_config user_config; 23562306a36Sopenharmony_ci struct i915_perf *perf = &i915->perf; 23662306a36Sopenharmony_ci struct i915_oa_config *oa_config; 23762306a36Sopenharmony_ci char uuid[UUID_STRING_LEN + 1]; 23862306a36Sopenharmony_ci u64 config_id; 23962306a36Sopenharmony_ci u32 flags, total_size; 24062306a36Sopenharmony_ci int ret; 24162306a36Sopenharmony_ci 24262306a36Sopenharmony_ci if (!perf->i915) 24362306a36Sopenharmony_ci return -ENODEV; 24462306a36Sopenharmony_ci 24562306a36Sopenharmony_ci total_size = 24662306a36Sopenharmony_ci sizeof(struct drm_i915_query_perf_config) + 24762306a36Sopenharmony_ci sizeof(struct drm_i915_perf_oa_config); 24862306a36Sopenharmony_ci 24962306a36Sopenharmony_ci if (query_item->length == 0) 25062306a36Sopenharmony_ci return total_size; 25162306a36Sopenharmony_ci 25262306a36Sopenharmony_ci if (query_item->length < total_size) { 25362306a36Sopenharmony_ci drm_dbg(&i915->drm, 25462306a36Sopenharmony_ci "Invalid query config data item size=%u expected=%u\n", 25562306a36Sopenharmony_ci query_item->length, total_size); 25662306a36Sopenharmony_ci return -EINVAL; 25762306a36Sopenharmony_ci } 25862306a36Sopenharmony_ci 25962306a36Sopenharmony_ci if (get_user(flags, &user_query_config_ptr->flags)) 26062306a36Sopenharmony_ci return -EFAULT; 26162306a36Sopenharmony_ci 26262306a36Sopenharmony_ci if (flags != 0) 26362306a36Sopenharmony_ci return -EINVAL; 26462306a36Sopenharmony_ci 26562306a36Sopenharmony_ci if (use_uuid) { 26662306a36Sopenharmony_ci struct i915_oa_config *tmp; 26762306a36Sopenharmony_ci int id; 26862306a36Sopenharmony_ci 26962306a36Sopenharmony_ci BUILD_BUG_ON(sizeof(user_query_config_ptr->uuid) >= sizeof(uuid)); 27062306a36Sopenharmony_ci 27162306a36Sopenharmony_ci memset(&uuid, 0, sizeof(uuid)); 27262306a36Sopenharmony_ci if (copy_from_user(uuid, user_query_config_ptr->uuid, 27362306a36Sopenharmony_ci sizeof(user_query_config_ptr->uuid))) 27462306a36Sopenharmony_ci return -EFAULT; 27562306a36Sopenharmony_ci 27662306a36Sopenharmony_ci oa_config = NULL; 27762306a36Sopenharmony_ci rcu_read_lock(); 27862306a36Sopenharmony_ci idr_for_each_entry(&perf->metrics_idr, tmp, id) { 27962306a36Sopenharmony_ci if (!strcmp(tmp->uuid, uuid)) { 28062306a36Sopenharmony_ci oa_config = i915_oa_config_get(tmp); 28162306a36Sopenharmony_ci break; 28262306a36Sopenharmony_ci } 28362306a36Sopenharmony_ci } 28462306a36Sopenharmony_ci rcu_read_unlock(); 28562306a36Sopenharmony_ci } else { 28662306a36Sopenharmony_ci if (get_user(config_id, &user_query_config_ptr->config)) 28762306a36Sopenharmony_ci return -EFAULT; 28862306a36Sopenharmony_ci 28962306a36Sopenharmony_ci oa_config = i915_perf_get_oa_config(perf, config_id); 29062306a36Sopenharmony_ci } 29162306a36Sopenharmony_ci if (!oa_config) 29262306a36Sopenharmony_ci return -ENOENT; 29362306a36Sopenharmony_ci 29462306a36Sopenharmony_ci if (copy_from_user(&user_config, user_config_ptr, sizeof(user_config))) { 29562306a36Sopenharmony_ci ret = -EFAULT; 29662306a36Sopenharmony_ci goto out; 29762306a36Sopenharmony_ci } 29862306a36Sopenharmony_ci 29962306a36Sopenharmony_ci ret = can_copy_perf_config_registers_or_number(user_config.n_boolean_regs, 30062306a36Sopenharmony_ci user_config.boolean_regs_ptr, 30162306a36Sopenharmony_ci oa_config->b_counter_regs_len); 30262306a36Sopenharmony_ci if (ret) 30362306a36Sopenharmony_ci goto out; 30462306a36Sopenharmony_ci 30562306a36Sopenharmony_ci ret = can_copy_perf_config_registers_or_number(user_config.n_flex_regs, 30662306a36Sopenharmony_ci user_config.flex_regs_ptr, 30762306a36Sopenharmony_ci oa_config->flex_regs_len); 30862306a36Sopenharmony_ci if (ret) 30962306a36Sopenharmony_ci goto out; 31062306a36Sopenharmony_ci 31162306a36Sopenharmony_ci ret = can_copy_perf_config_registers_or_number(user_config.n_mux_regs, 31262306a36Sopenharmony_ci user_config.mux_regs_ptr, 31362306a36Sopenharmony_ci oa_config->mux_regs_len); 31462306a36Sopenharmony_ci if (ret) 31562306a36Sopenharmony_ci goto out; 31662306a36Sopenharmony_ci 31762306a36Sopenharmony_ci ret = copy_perf_config_registers_or_number(oa_config->b_counter_regs, 31862306a36Sopenharmony_ci oa_config->b_counter_regs_len, 31962306a36Sopenharmony_ci user_config.boolean_regs_ptr, 32062306a36Sopenharmony_ci &user_config.n_boolean_regs); 32162306a36Sopenharmony_ci if (ret) 32262306a36Sopenharmony_ci goto out; 32362306a36Sopenharmony_ci 32462306a36Sopenharmony_ci ret = copy_perf_config_registers_or_number(oa_config->flex_regs, 32562306a36Sopenharmony_ci oa_config->flex_regs_len, 32662306a36Sopenharmony_ci user_config.flex_regs_ptr, 32762306a36Sopenharmony_ci &user_config.n_flex_regs); 32862306a36Sopenharmony_ci if (ret) 32962306a36Sopenharmony_ci goto out; 33062306a36Sopenharmony_ci 33162306a36Sopenharmony_ci ret = copy_perf_config_registers_or_number(oa_config->mux_regs, 33262306a36Sopenharmony_ci oa_config->mux_regs_len, 33362306a36Sopenharmony_ci user_config.mux_regs_ptr, 33462306a36Sopenharmony_ci &user_config.n_mux_regs); 33562306a36Sopenharmony_ci if (ret) 33662306a36Sopenharmony_ci goto out; 33762306a36Sopenharmony_ci 33862306a36Sopenharmony_ci memcpy(user_config.uuid, oa_config->uuid, sizeof(user_config.uuid)); 33962306a36Sopenharmony_ci 34062306a36Sopenharmony_ci if (copy_to_user(user_config_ptr, &user_config, sizeof(user_config))) { 34162306a36Sopenharmony_ci ret = -EFAULT; 34262306a36Sopenharmony_ci goto out; 34362306a36Sopenharmony_ci } 34462306a36Sopenharmony_ci 34562306a36Sopenharmony_ci ret = total_size; 34662306a36Sopenharmony_ci 34762306a36Sopenharmony_ciout: 34862306a36Sopenharmony_ci i915_oa_config_put(oa_config); 34962306a36Sopenharmony_ci return ret; 35062306a36Sopenharmony_ci} 35162306a36Sopenharmony_ci 35262306a36Sopenharmony_cistatic size_t sizeof_perf_config_list(size_t count) 35362306a36Sopenharmony_ci{ 35462306a36Sopenharmony_ci return sizeof(struct drm_i915_query_perf_config) + sizeof(u64) * count; 35562306a36Sopenharmony_ci} 35662306a36Sopenharmony_ci 35762306a36Sopenharmony_cistatic size_t sizeof_perf_metrics(struct i915_perf *perf) 35862306a36Sopenharmony_ci{ 35962306a36Sopenharmony_ci struct i915_oa_config *tmp; 36062306a36Sopenharmony_ci size_t i; 36162306a36Sopenharmony_ci int id; 36262306a36Sopenharmony_ci 36362306a36Sopenharmony_ci i = 1; 36462306a36Sopenharmony_ci rcu_read_lock(); 36562306a36Sopenharmony_ci idr_for_each_entry(&perf->metrics_idr, tmp, id) 36662306a36Sopenharmony_ci i++; 36762306a36Sopenharmony_ci rcu_read_unlock(); 36862306a36Sopenharmony_ci 36962306a36Sopenharmony_ci return sizeof_perf_config_list(i); 37062306a36Sopenharmony_ci} 37162306a36Sopenharmony_ci 37262306a36Sopenharmony_cistatic int query_perf_config_list(struct drm_i915_private *i915, 37362306a36Sopenharmony_ci struct drm_i915_query_item *query_item) 37462306a36Sopenharmony_ci{ 37562306a36Sopenharmony_ci struct drm_i915_query_perf_config __user *user_query_config_ptr = 37662306a36Sopenharmony_ci u64_to_user_ptr(query_item->data_ptr); 37762306a36Sopenharmony_ci struct i915_perf *perf = &i915->perf; 37862306a36Sopenharmony_ci u64 *oa_config_ids = NULL; 37962306a36Sopenharmony_ci int alloc, n_configs; 38062306a36Sopenharmony_ci u32 flags; 38162306a36Sopenharmony_ci int ret; 38262306a36Sopenharmony_ci 38362306a36Sopenharmony_ci if (!perf->i915) 38462306a36Sopenharmony_ci return -ENODEV; 38562306a36Sopenharmony_ci 38662306a36Sopenharmony_ci if (query_item->length == 0) 38762306a36Sopenharmony_ci return sizeof_perf_metrics(perf); 38862306a36Sopenharmony_ci 38962306a36Sopenharmony_ci if (get_user(flags, &user_query_config_ptr->flags)) 39062306a36Sopenharmony_ci return -EFAULT; 39162306a36Sopenharmony_ci 39262306a36Sopenharmony_ci if (flags != 0) 39362306a36Sopenharmony_ci return -EINVAL; 39462306a36Sopenharmony_ci 39562306a36Sopenharmony_ci n_configs = 1; 39662306a36Sopenharmony_ci do { 39762306a36Sopenharmony_ci struct i915_oa_config *tmp; 39862306a36Sopenharmony_ci u64 *ids; 39962306a36Sopenharmony_ci int id; 40062306a36Sopenharmony_ci 40162306a36Sopenharmony_ci ids = krealloc(oa_config_ids, 40262306a36Sopenharmony_ci n_configs * sizeof(*oa_config_ids), 40362306a36Sopenharmony_ci GFP_KERNEL); 40462306a36Sopenharmony_ci if (!ids) 40562306a36Sopenharmony_ci return -ENOMEM; 40662306a36Sopenharmony_ci 40762306a36Sopenharmony_ci alloc = fetch_and_zero(&n_configs); 40862306a36Sopenharmony_ci 40962306a36Sopenharmony_ci ids[n_configs++] = 1ull; /* reserved for test_config */ 41062306a36Sopenharmony_ci rcu_read_lock(); 41162306a36Sopenharmony_ci idr_for_each_entry(&perf->metrics_idr, tmp, id) { 41262306a36Sopenharmony_ci if (n_configs < alloc) 41362306a36Sopenharmony_ci ids[n_configs] = id; 41462306a36Sopenharmony_ci n_configs++; 41562306a36Sopenharmony_ci } 41662306a36Sopenharmony_ci rcu_read_unlock(); 41762306a36Sopenharmony_ci 41862306a36Sopenharmony_ci oa_config_ids = ids; 41962306a36Sopenharmony_ci } while (n_configs > alloc); 42062306a36Sopenharmony_ci 42162306a36Sopenharmony_ci if (query_item->length < sizeof_perf_config_list(n_configs)) { 42262306a36Sopenharmony_ci drm_dbg(&i915->drm, 42362306a36Sopenharmony_ci "Invalid query config list item size=%u expected=%zu\n", 42462306a36Sopenharmony_ci query_item->length, 42562306a36Sopenharmony_ci sizeof_perf_config_list(n_configs)); 42662306a36Sopenharmony_ci kfree(oa_config_ids); 42762306a36Sopenharmony_ci return -EINVAL; 42862306a36Sopenharmony_ci } 42962306a36Sopenharmony_ci 43062306a36Sopenharmony_ci if (put_user(n_configs, &user_query_config_ptr->config)) { 43162306a36Sopenharmony_ci kfree(oa_config_ids); 43262306a36Sopenharmony_ci return -EFAULT; 43362306a36Sopenharmony_ci } 43462306a36Sopenharmony_ci 43562306a36Sopenharmony_ci ret = copy_to_user(user_query_config_ptr + 1, 43662306a36Sopenharmony_ci oa_config_ids, 43762306a36Sopenharmony_ci n_configs * sizeof(*oa_config_ids)); 43862306a36Sopenharmony_ci kfree(oa_config_ids); 43962306a36Sopenharmony_ci if (ret) 44062306a36Sopenharmony_ci return -EFAULT; 44162306a36Sopenharmony_ci 44262306a36Sopenharmony_ci return sizeof_perf_config_list(n_configs); 44362306a36Sopenharmony_ci} 44462306a36Sopenharmony_ci 44562306a36Sopenharmony_cistatic int query_perf_config(struct drm_i915_private *i915, 44662306a36Sopenharmony_ci struct drm_i915_query_item *query_item) 44762306a36Sopenharmony_ci{ 44862306a36Sopenharmony_ci switch (query_item->flags) { 44962306a36Sopenharmony_ci case DRM_I915_QUERY_PERF_CONFIG_LIST: 45062306a36Sopenharmony_ci return query_perf_config_list(i915, query_item); 45162306a36Sopenharmony_ci case DRM_I915_QUERY_PERF_CONFIG_DATA_FOR_UUID: 45262306a36Sopenharmony_ci return query_perf_config_data(i915, query_item, true); 45362306a36Sopenharmony_ci case DRM_I915_QUERY_PERF_CONFIG_DATA_FOR_ID: 45462306a36Sopenharmony_ci return query_perf_config_data(i915, query_item, false); 45562306a36Sopenharmony_ci default: 45662306a36Sopenharmony_ci return -EINVAL; 45762306a36Sopenharmony_ci } 45862306a36Sopenharmony_ci} 45962306a36Sopenharmony_ci 46062306a36Sopenharmony_cistatic int query_memregion_info(struct drm_i915_private *i915, 46162306a36Sopenharmony_ci struct drm_i915_query_item *query_item) 46262306a36Sopenharmony_ci{ 46362306a36Sopenharmony_ci struct drm_i915_query_memory_regions __user *query_ptr = 46462306a36Sopenharmony_ci u64_to_user_ptr(query_item->data_ptr); 46562306a36Sopenharmony_ci struct drm_i915_memory_region_info __user *info_ptr = 46662306a36Sopenharmony_ci &query_ptr->regions[0]; 46762306a36Sopenharmony_ci struct drm_i915_memory_region_info info = { }; 46862306a36Sopenharmony_ci struct drm_i915_query_memory_regions query; 46962306a36Sopenharmony_ci struct intel_memory_region *mr; 47062306a36Sopenharmony_ci u32 total_length; 47162306a36Sopenharmony_ci int ret, id, i; 47262306a36Sopenharmony_ci 47362306a36Sopenharmony_ci if (query_item->flags != 0) 47462306a36Sopenharmony_ci return -EINVAL; 47562306a36Sopenharmony_ci 47662306a36Sopenharmony_ci total_length = sizeof(query); 47762306a36Sopenharmony_ci for_each_memory_region(mr, i915, id) { 47862306a36Sopenharmony_ci if (mr->private) 47962306a36Sopenharmony_ci continue; 48062306a36Sopenharmony_ci 48162306a36Sopenharmony_ci total_length += sizeof(info); 48262306a36Sopenharmony_ci } 48362306a36Sopenharmony_ci 48462306a36Sopenharmony_ci ret = copy_query_item(&query, sizeof(query), total_length, query_item); 48562306a36Sopenharmony_ci if (ret != 0) 48662306a36Sopenharmony_ci return ret; 48762306a36Sopenharmony_ci 48862306a36Sopenharmony_ci if (query.num_regions) 48962306a36Sopenharmony_ci return -EINVAL; 49062306a36Sopenharmony_ci 49162306a36Sopenharmony_ci for (i = 0; i < ARRAY_SIZE(query.rsvd); i++) { 49262306a36Sopenharmony_ci if (query.rsvd[i]) 49362306a36Sopenharmony_ci return -EINVAL; 49462306a36Sopenharmony_ci } 49562306a36Sopenharmony_ci 49662306a36Sopenharmony_ci for_each_memory_region(mr, i915, id) { 49762306a36Sopenharmony_ci if (mr->private) 49862306a36Sopenharmony_ci continue; 49962306a36Sopenharmony_ci 50062306a36Sopenharmony_ci info.region.memory_class = mr->type; 50162306a36Sopenharmony_ci info.region.memory_instance = mr->instance; 50262306a36Sopenharmony_ci info.probed_size = mr->total; 50362306a36Sopenharmony_ci 50462306a36Sopenharmony_ci if (mr->type == INTEL_MEMORY_LOCAL) 50562306a36Sopenharmony_ci info.probed_cpu_visible_size = mr->io_size; 50662306a36Sopenharmony_ci else 50762306a36Sopenharmony_ci info.probed_cpu_visible_size = mr->total; 50862306a36Sopenharmony_ci 50962306a36Sopenharmony_ci if (perfmon_capable()) { 51062306a36Sopenharmony_ci intel_memory_region_avail(mr, 51162306a36Sopenharmony_ci &info.unallocated_size, 51262306a36Sopenharmony_ci &info.unallocated_cpu_visible_size); 51362306a36Sopenharmony_ci } else { 51462306a36Sopenharmony_ci info.unallocated_size = info.probed_size; 51562306a36Sopenharmony_ci info.unallocated_cpu_visible_size = 51662306a36Sopenharmony_ci info.probed_cpu_visible_size; 51762306a36Sopenharmony_ci } 51862306a36Sopenharmony_ci 51962306a36Sopenharmony_ci if (__copy_to_user(info_ptr, &info, sizeof(info))) 52062306a36Sopenharmony_ci return -EFAULT; 52162306a36Sopenharmony_ci 52262306a36Sopenharmony_ci query.num_regions++; 52362306a36Sopenharmony_ci info_ptr++; 52462306a36Sopenharmony_ci } 52562306a36Sopenharmony_ci 52662306a36Sopenharmony_ci if (__copy_to_user(query_ptr, &query, sizeof(query))) 52762306a36Sopenharmony_ci return -EFAULT; 52862306a36Sopenharmony_ci 52962306a36Sopenharmony_ci return total_length; 53062306a36Sopenharmony_ci} 53162306a36Sopenharmony_ci 53262306a36Sopenharmony_cistatic int query_hwconfig_blob(struct drm_i915_private *i915, 53362306a36Sopenharmony_ci struct drm_i915_query_item *query_item) 53462306a36Sopenharmony_ci{ 53562306a36Sopenharmony_ci struct intel_gt *gt = to_gt(i915); 53662306a36Sopenharmony_ci struct intel_hwconfig *hwconfig = >->info.hwconfig; 53762306a36Sopenharmony_ci 53862306a36Sopenharmony_ci if (!hwconfig->size || !hwconfig->ptr) 53962306a36Sopenharmony_ci return -ENODEV; 54062306a36Sopenharmony_ci 54162306a36Sopenharmony_ci if (query_item->length == 0) 54262306a36Sopenharmony_ci return hwconfig->size; 54362306a36Sopenharmony_ci 54462306a36Sopenharmony_ci if (query_item->length < hwconfig->size) 54562306a36Sopenharmony_ci return -EINVAL; 54662306a36Sopenharmony_ci 54762306a36Sopenharmony_ci if (copy_to_user(u64_to_user_ptr(query_item->data_ptr), 54862306a36Sopenharmony_ci hwconfig->ptr, hwconfig->size)) 54962306a36Sopenharmony_ci return -EFAULT; 55062306a36Sopenharmony_ci 55162306a36Sopenharmony_ci return hwconfig->size; 55262306a36Sopenharmony_ci} 55362306a36Sopenharmony_ci 55462306a36Sopenharmony_cistatic int (* const i915_query_funcs[])(struct drm_i915_private *dev_priv, 55562306a36Sopenharmony_ci struct drm_i915_query_item *query_item) = { 55662306a36Sopenharmony_ci query_topology_info, 55762306a36Sopenharmony_ci query_engine_info, 55862306a36Sopenharmony_ci query_perf_config, 55962306a36Sopenharmony_ci query_memregion_info, 56062306a36Sopenharmony_ci query_hwconfig_blob, 56162306a36Sopenharmony_ci query_geometry_subslices, 56262306a36Sopenharmony_ci}; 56362306a36Sopenharmony_ci 56462306a36Sopenharmony_ciint i915_query_ioctl(struct drm_device *dev, void *data, struct drm_file *file) 56562306a36Sopenharmony_ci{ 56662306a36Sopenharmony_ci struct drm_i915_private *dev_priv = to_i915(dev); 56762306a36Sopenharmony_ci struct drm_i915_query *args = data; 56862306a36Sopenharmony_ci struct drm_i915_query_item __user *user_item_ptr = 56962306a36Sopenharmony_ci u64_to_user_ptr(args->items_ptr); 57062306a36Sopenharmony_ci u32 i; 57162306a36Sopenharmony_ci 57262306a36Sopenharmony_ci if (args->flags != 0) 57362306a36Sopenharmony_ci return -EINVAL; 57462306a36Sopenharmony_ci 57562306a36Sopenharmony_ci for (i = 0; i < args->num_items; i++, user_item_ptr++) { 57662306a36Sopenharmony_ci struct drm_i915_query_item item; 57762306a36Sopenharmony_ci unsigned long func_idx; 57862306a36Sopenharmony_ci int ret; 57962306a36Sopenharmony_ci 58062306a36Sopenharmony_ci if (copy_from_user(&item, user_item_ptr, sizeof(item))) 58162306a36Sopenharmony_ci return -EFAULT; 58262306a36Sopenharmony_ci 58362306a36Sopenharmony_ci if (item.query_id == 0) 58462306a36Sopenharmony_ci return -EINVAL; 58562306a36Sopenharmony_ci 58662306a36Sopenharmony_ci if (overflows_type(item.query_id - 1, unsigned long)) 58762306a36Sopenharmony_ci return -EINVAL; 58862306a36Sopenharmony_ci 58962306a36Sopenharmony_ci func_idx = item.query_id - 1; 59062306a36Sopenharmony_ci 59162306a36Sopenharmony_ci ret = -EINVAL; 59262306a36Sopenharmony_ci if (func_idx < ARRAY_SIZE(i915_query_funcs)) { 59362306a36Sopenharmony_ci func_idx = array_index_nospec(func_idx, 59462306a36Sopenharmony_ci ARRAY_SIZE(i915_query_funcs)); 59562306a36Sopenharmony_ci ret = i915_query_funcs[func_idx](dev_priv, &item); 59662306a36Sopenharmony_ci } 59762306a36Sopenharmony_ci 59862306a36Sopenharmony_ci /* Only write the length back to userspace if they differ. */ 59962306a36Sopenharmony_ci if (ret != item.length && put_user(ret, &user_item_ptr->length)) 60062306a36Sopenharmony_ci return -EFAULT; 60162306a36Sopenharmony_ci } 60262306a36Sopenharmony_ci 60362306a36Sopenharmony_ci return 0; 60462306a36Sopenharmony_ci} 605