drm/i915/cmdparser: Improve hash function
The existing code's hashfunction is very suboptimal (most 3D commands use the same bucket degrading the hash to a long list). The code even acknowledge that the issue was known and the fix simple: /* * If we attempt to generate a perfect hash, we should be able to look at bits * 31:29 of a command from a batch buffer and use the full mask for that * client. The existing INSTR_CLIENT_MASK/SHIFT defines can be used for this. */ Signed-off-by: Chris Wilson <chris@chris-wilson.co.uk> Reviewed-by: Matthew Auld <matthew.auld@intel.com> Link: http://patchwork.freedesktop.org/patch/msgid/20160818161718.27187-35-chris@chris-wilson.co.uk
This commit is contained in:
Родитель
ed13033f02
Коммит
d6a4ead7a3
|
@ -86,24 +86,24 @@
|
||||||
* general bitmasking mechanism.
|
* general bitmasking mechanism.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#define STD_MI_OPCODE_MASK 0xFF800000
|
#define STD_MI_OPCODE_SHIFT (32 - 9)
|
||||||
#define STD_3D_OPCODE_MASK 0xFFFF0000
|
#define STD_3D_OPCODE_SHIFT (32 - 16)
|
||||||
#define STD_2D_OPCODE_MASK 0xFFC00000
|
#define STD_2D_OPCODE_SHIFT (32 - 10)
|
||||||
#define STD_MFX_OPCODE_MASK 0xFFFF0000
|
#define STD_MFX_OPCODE_SHIFT (32 - 16)
|
||||||
|
|
||||||
#define CMD(op, opm, f, lm, fl, ...) \
|
#define CMD(op, opm, f, lm, fl, ...) \
|
||||||
{ \
|
{ \
|
||||||
.flags = (fl) | ((f) ? CMD_DESC_FIXED : 0), \
|
.flags = (fl) | ((f) ? CMD_DESC_FIXED : 0), \
|
||||||
.cmd = { (op), (opm) }, \
|
.cmd = { (op), ~0u << (opm) }, \
|
||||||
.length = { (lm) }, \
|
.length = { (lm) }, \
|
||||||
__VA_ARGS__ \
|
__VA_ARGS__ \
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Convenience macros to compress the tables */
|
/* Convenience macros to compress the tables */
|
||||||
#define SMI STD_MI_OPCODE_MASK
|
#define SMI STD_MI_OPCODE_SHIFT
|
||||||
#define S3D STD_3D_OPCODE_MASK
|
#define S3D STD_3D_OPCODE_SHIFT
|
||||||
#define S2D STD_2D_OPCODE_MASK
|
#define S2D STD_2D_OPCODE_SHIFT
|
||||||
#define SMFX STD_MFX_OPCODE_MASK
|
#define SMFX STD_MFX_OPCODE_SHIFT
|
||||||
#define F true
|
#define F true
|
||||||
#define S CMD_DESC_SKIP
|
#define S CMD_DESC_SKIP
|
||||||
#define R CMD_DESC_REJECT
|
#define R CMD_DESC_REJECT
|
||||||
|
@ -696,12 +696,26 @@ struct cmd_node {
|
||||||
* non-opcode bits being set. But if we don't include those bits, some 3D
|
* non-opcode bits being set. But if we don't include those bits, some 3D
|
||||||
* commands may hash to the same bucket due to not including opcode bits that
|
* commands may hash to the same bucket due to not including opcode bits that
|
||||||
* make the command unique. For now, we will risk hashing to the same bucket.
|
* make the command unique. For now, we will risk hashing to the same bucket.
|
||||||
*
|
|
||||||
* If we attempt to generate a perfect hash, we should be able to look at bits
|
|
||||||
* 31:29 of a command from a batch buffer and use the full mask for that
|
|
||||||
* client. The existing INSTR_CLIENT_MASK/SHIFT defines can be used for this.
|
|
||||||
*/
|
*/
|
||||||
#define CMD_HASH_MASK STD_MI_OPCODE_MASK
|
static inline u32 cmd_header_key(u32 x)
|
||||||
|
{
|
||||||
|
u32 shift;
|
||||||
|
|
||||||
|
switch (x >> INSTR_CLIENT_SHIFT) {
|
||||||
|
default:
|
||||||
|
case INSTR_MI_CLIENT:
|
||||||
|
shift = STD_MI_OPCODE_SHIFT;
|
||||||
|
break;
|
||||||
|
case INSTR_RC_CLIENT:
|
||||||
|
shift = STD_3D_OPCODE_SHIFT;
|
||||||
|
break;
|
||||||
|
case INSTR_BC_CLIENT:
|
||||||
|
shift = STD_2D_OPCODE_SHIFT;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return x >> shift;
|
||||||
|
}
|
||||||
|
|
||||||
static int init_hash_table(struct intel_engine_cs *engine,
|
static int init_hash_table(struct intel_engine_cs *engine,
|
||||||
const struct drm_i915_cmd_table *cmd_tables,
|
const struct drm_i915_cmd_table *cmd_tables,
|
||||||
|
@ -725,7 +739,7 @@ static int init_hash_table(struct intel_engine_cs *engine,
|
||||||
|
|
||||||
desc_node->desc = desc;
|
desc_node->desc = desc;
|
||||||
hash_add(engine->cmd_hash, &desc_node->node,
|
hash_add(engine->cmd_hash, &desc_node->node,
|
||||||
desc->cmd.value & CMD_HASH_MASK);
|
cmd_header_key(desc->cmd.value));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -859,12 +873,9 @@ find_cmd_in_table(struct intel_engine_cs *engine,
|
||||||
struct cmd_node *desc_node;
|
struct cmd_node *desc_node;
|
||||||
|
|
||||||
hash_for_each_possible(engine->cmd_hash, desc_node, node,
|
hash_for_each_possible(engine->cmd_hash, desc_node, node,
|
||||||
cmd_header & CMD_HASH_MASK) {
|
cmd_header_key(cmd_header)) {
|
||||||
const struct drm_i915_cmd_descriptor *desc = desc_node->desc;
|
const struct drm_i915_cmd_descriptor *desc = desc_node->desc;
|
||||||
u32 masked_cmd = desc->cmd.mask & cmd_header;
|
if (((cmd_header ^ desc->cmd.value) & desc->cmd.mask) == 0)
|
||||||
u32 masked_value = desc->cmd.value & desc->cmd.mask;
|
|
||||||
|
|
||||||
if (masked_cmd == masked_value)
|
|
||||||
return desc;
|
return desc;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Загрузка…
Ссылка в новой задаче