Commit a47ac10e authored by Dmitry Osipenko's avatar Dmitry Osipenko Committed by Thierry Reding

gpu: host1x: Check waits in the firewall

Check waits in the firewall in a way it is done for relocations.
Signed-off-by: default avatarDmitry Osipenko <digetx@gmail.com>
Reviewed-by: default avatarMikko Perttunen <mperttunen@nvidia.com>
Reviewed-by: default avatarErik Faye-Lund <kusmabite@gmail.com>
Signed-off-by: default avatarThierry Reding <treding@nvidia.com>
parent a2b78b0d
...@@ -31,6 +31,8 @@ ...@@ -31,6 +31,8 @@
#include "job.h" #include "job.h"
#include "syncpt.h" #include "syncpt.h"
#define HOST1X_WAIT_SYNCPT_OFFSET 0x8
struct host1x_job *host1x_job_alloc(struct host1x_channel *ch, struct host1x_job *host1x_job_alloc(struct host1x_channel *ch,
u32 num_cmdbufs, u32 num_relocs, u32 num_cmdbufs, u32 num_relocs,
u32 num_waitchks) u32 num_waitchks)
...@@ -337,6 +339,17 @@ static bool check_reloc(struct host1x_reloc *reloc, struct host1x_bo *cmdbuf, ...@@ -337,6 +339,17 @@ static bool check_reloc(struct host1x_reloc *reloc, struct host1x_bo *cmdbuf,
return true; return true;
} }
static bool check_wait(struct host1x_waitchk *wait, struct host1x_bo *cmdbuf,
unsigned int offset)
{
offset *= sizeof(u32);
if (wait->bo != cmdbuf || wait->offset != offset)
return false;
return true;
}
struct host1x_firewall { struct host1x_firewall {
struct host1x_job *job; struct host1x_job *job;
struct device *dev; struct device *dev;
...@@ -344,6 +357,9 @@ struct host1x_firewall { ...@@ -344,6 +357,9 @@ struct host1x_firewall {
unsigned int num_relocs; unsigned int num_relocs;
struct host1x_reloc *reloc; struct host1x_reloc *reloc;
unsigned int num_waitchks;
struct host1x_waitchk *waitchk;
struct host1x_bo *cmdbuf; struct host1x_bo *cmdbuf;
unsigned int offset; unsigned int offset;
...@@ -370,6 +386,20 @@ static int check_register(struct host1x_firewall *fw, unsigned long offset) ...@@ -370,6 +386,20 @@ static int check_register(struct host1x_firewall *fw, unsigned long offset)
fw->reloc++; fw->reloc++;
} }
if (offset == HOST1X_WAIT_SYNCPT_OFFSET) {
if (fw->class != HOST1X_CLASS_HOST1X)
return -EINVAL;
if (!fw->num_waitchks)
return -EINVAL;
if (!check_wait(fw->waitchk, fw->cmdbuf, fw->offset))
return -EINVAL;
fw->num_waitchks--;
fw->waitchk++;
}
return 0; return 0;
} }
...@@ -534,6 +564,8 @@ static inline int copy_gathers(struct host1x_job *job, struct device *dev) ...@@ -534,6 +564,8 @@ static inline int copy_gathers(struct host1x_job *job, struct device *dev)
fw.dev = dev; fw.dev = dev;
fw.reloc = job->relocarray; fw.reloc = job->relocarray;
fw.num_relocs = job->num_relocs; fw.num_relocs = job->num_relocs;
fw.waitchk = job->waitchk;
fw.num_waitchks = job->num_waitchk;
fw.class = job->class; fw.class = job->class;
for (i = 0; i < job->num_gathers; i++) { for (i = 0; i < job->num_gathers; i++) {
...@@ -572,8 +604,8 @@ static inline int copy_gathers(struct host1x_job *job, struct device *dev) ...@@ -572,8 +604,8 @@ static inline int copy_gathers(struct host1x_job *job, struct device *dev)
offset += g->words * sizeof(u32); offset += g->words * sizeof(u32);
} }
/* No relocs should remain at this point */ /* No relocs and waitchks should remain at this point */
if (fw.num_relocs) if (fw.num_relocs || fw.num_waitchks)
return -EINVAL; return -EINVAL;
return 0; return 0;
......
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