Commit e5b69da6 authored by Grazvydas Ignotas's avatar Grazvydas Ignotas Committed by Alex Deucher

drm/radeon: split evergreen_cs_check_reg

evergreen_cs_check_reg() is a large function and gcc doesn't want to
inline it. It has a quick check for reg_safe_bm[] to see if register
needs special handling, which often results in early exit. However
because the function is large, it has a long prologue/epilogue to
save/restore all the callee-save registers which according to perf is
taking significant amount of time. To avoid this, we can reuse
evergreen_is_safe_reg() to do the early check directly in register loop.
Reviewed-by: default avatarDave Airlie <airlied@redhat.com>
Signed-off-by: default avatarGrazvydas Ignotas <notasas@gmail.com>
Signed-off-by: default avatarAlex Deucher <alexander.deucher@amd.com>
parent 4a985353
...@@ -1086,30 +1086,18 @@ static int evergreen_cs_parse_packet0(struct radeon_cs_parser *p, ...@@ -1086,30 +1086,18 @@ static int evergreen_cs_parse_packet0(struct radeon_cs_parser *p,
} }
/** /**
* evergreen_cs_check_reg() - check if register is authorized or not * evergreen_cs_handle_reg() - process registers that need special handling.
* @parser: parser structure holding parsing context * @parser: parser structure holding parsing context
* @reg: register we are testing * @reg: register we are testing
* @idx: index into the cs buffer * @idx: index into the cs buffer
*
* This function will test against evergreen_reg_safe_bm and return 0
* if register is safe. If register is not flag as safe this function
* will test it against a list of register needind special handling.
*/ */
static int evergreen_cs_check_reg(struct radeon_cs_parser *p, u32 reg, u32 idx) static int evergreen_cs_handle_reg(struct radeon_cs_parser *p, u32 reg, u32 idx)
{ {
struct evergreen_cs_track *track = (struct evergreen_cs_track *)p->track; struct evergreen_cs_track *track = (struct evergreen_cs_track *)p->track;
struct radeon_bo_list *reloc; struct radeon_bo_list *reloc;
u32 m, i, tmp, *ib; u32 tmp, *ib;
int r; int r;
i = (reg >> 7);
if (unlikely(i >= REG_SAFE_BM_SIZE)) {
dev_warn(p->dev, "forbidden register 0x%08x at %d\n", reg, idx);
return -EINVAL;
}
m = 1 << ((reg >> 2) & 31);
if (!(track->reg_safe_bm[i] & m))
return 0;
ib = p->ib.ptr; ib = p->ib.ptr;
switch (reg) { switch (reg) {
/* force following reg to 0 in an attempt to disable out buffer /* force following reg to 0 in an attempt to disable out buffer
...@@ -1756,20 +1744,27 @@ static int evergreen_cs_check_reg(struct radeon_cs_parser *p, u32 reg, u32 idx) ...@@ -1756,20 +1744,27 @@ static int evergreen_cs_check_reg(struct radeon_cs_parser *p, u32 reg, u32 idx)
return 0; return 0;
} }
static bool evergreen_is_safe_reg(struct radeon_cs_parser *p, u32 reg, u32 idx) /**
* evergreen_is_safe_reg() - check if register is authorized or not
* @parser: parser structure holding parsing context
* @reg: register we are testing
*
* This function will test against reg_safe_bm and return true
* if register is safe or false otherwise.
*/
static inline bool evergreen_is_safe_reg(struct radeon_cs_parser *p, u32 reg)
{ {
struct evergreen_cs_track *track = p->track; struct evergreen_cs_track *track = p->track;
u32 m, i; u32 m, i;
i = (reg >> 7); i = (reg >> 7);
if (unlikely(i >= REG_SAFE_BM_SIZE)) { if (unlikely(i >= REG_SAFE_BM_SIZE)) {
dev_warn(p->dev, "forbidden register 0x%08x at %d\n", reg, idx);
return false; return false;
} }
m = 1 << ((reg >> 2) & 31); m = 1 << ((reg >> 2) & 31);
if (!(track->reg_safe_bm[i] & m)) if (!(track->reg_safe_bm[i] & m))
return true; return true;
dev_warn(p->dev, "forbidden register 0x%08x at %d\n", reg, idx);
return false; return false;
} }
...@@ -2306,7 +2301,9 @@ static int evergreen_packet3_check(struct radeon_cs_parser *p, ...@@ -2306,7 +2301,9 @@ static int evergreen_packet3_check(struct radeon_cs_parser *p,
} }
for (i = 0; i < pkt->count; i++) { for (i = 0; i < pkt->count; i++) {
reg = start_reg + (4 * i); reg = start_reg + (4 * i);
r = evergreen_cs_check_reg(p, reg, idx+1+i); if (evergreen_is_safe_reg(p, reg))
continue;
r = evergreen_cs_handle_reg(p, reg, idx + 1 + i);
if (r) if (r)
return r; return r;
} }
...@@ -2322,7 +2319,9 @@ static int evergreen_packet3_check(struct radeon_cs_parser *p, ...@@ -2322,7 +2319,9 @@ static int evergreen_packet3_check(struct radeon_cs_parser *p,
} }
for (i = 0; i < pkt->count; i++) { for (i = 0; i < pkt->count; i++) {
reg = start_reg + (4 * i); reg = start_reg + (4 * i);
r = evergreen_cs_check_reg(p, reg, idx+1+i); if (evergreen_is_safe_reg(p, reg))
continue;
r = evergreen_cs_handle_reg(p, reg, idx + 1 + i);
if (r) if (r)
return r; return r;
} }
...@@ -2577,9 +2576,12 @@ static int evergreen_packet3_check(struct radeon_cs_parser *p, ...@@ -2577,9 +2576,12 @@ static int evergreen_packet3_check(struct radeon_cs_parser *p,
} else { } else {
/* SRC is a reg. */ /* SRC is a reg. */
reg = radeon_get_ib_value(p, idx+1) << 2; reg = radeon_get_ib_value(p, idx+1) << 2;
if (!evergreen_is_safe_reg(p, reg, idx+1)) if (!evergreen_is_safe_reg(p, reg)) {
dev_warn(p->dev, "forbidden register 0x%08x at %d\n",
reg, idx + 1);
return -EINVAL; return -EINVAL;
} }
}
if (idx_value & 0x2) { if (idx_value & 0x2) {
u64 offset; u64 offset;
/* DST is memory. */ /* DST is memory. */
...@@ -2601,9 +2603,12 @@ static int evergreen_packet3_check(struct radeon_cs_parser *p, ...@@ -2601,9 +2603,12 @@ static int evergreen_packet3_check(struct radeon_cs_parser *p,
} else { } else {
/* DST is a reg. */ /* DST is a reg. */
reg = radeon_get_ib_value(p, idx+3) << 2; reg = radeon_get_ib_value(p, idx+3) << 2;
if (!evergreen_is_safe_reg(p, reg, idx+3)) if (!evergreen_is_safe_reg(p, reg)) {
dev_warn(p->dev, "forbidden register 0x%08x at %d\n",
reg, idx + 3);
return -EINVAL; return -EINVAL;
} }
}
break; break;
case PACKET3_NOP: case PACKET3_NOP:
break; break;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment