mirror of
git://nv-tegra.nvidia.com/linux-nv-oot.git
synced 2025-12-22 17:25:35 +03:00
drm/tegra: Update to UAPI v5
Update the tegra-drm driver to the 'Host1x/Tegra UAPI v5' series [0]. This fixes a few minor bugs found in the previous series. [0] https://patchwork.ozlabs.org/project/linux-tegra/list/?series=223684 Bug 200687525 Change-Id: I270016756b6b689c1fada208896cee81223e7042 Signed-off-by: Jon Hunter <jonathanh@nvidia.com> Reviewed-on: https://git-master.nvidia.com/r/c/linux-nvidia/+/2469984 Tested-by: mobile promotions <svcmobile_promotions@nvidia.com> Reviewed-by: Mikko Perttunen <mperttunen@nvidia.com> Reviewed-by: mobile promotions <svcmobile_promotions@nvidia.com> GVS: Gerrit_Virtual_Submit
This commit is contained in:
committed by
Laxman Dewangan
parent
d088bfa37d
commit
d3ea27d381
@@ -325,9 +325,6 @@ int tegra_drm_submit(struct tegra_drm_context *context,
|
|||||||
args->fence = job->syncpt_end;
|
args->fence = job->syncpt_end;
|
||||||
|
|
||||||
fail:
|
fail:
|
||||||
if (sp)
|
|
||||||
host1x_syncpt_put(sp);
|
|
||||||
|
|
||||||
while (num_refs--)
|
while (num_refs--)
|
||||||
drm_gem_object_put(refs[num_refs]);
|
drm_gem_object_put(refs[num_refs]);
|
||||||
|
|
||||||
@@ -731,6 +728,7 @@ static int tegra_gem_get_flags(struct drm_device *drm, void *data,
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
static const struct drm_ioctl_desc tegra_drm_ioctls[] = {
|
static const struct drm_ioctl_desc tegra_drm_ioctls[] = {
|
||||||
|
#ifdef CONFIG_DRM_TEGRA_STAGING
|
||||||
DRM_IOCTL_DEF_DRV(TEGRA_CHANNEL_OPEN, tegra_drm_ioctl_channel_open,
|
DRM_IOCTL_DEF_DRV(TEGRA_CHANNEL_OPEN, tegra_drm_ioctl_channel_open,
|
||||||
DRM_RENDER_ALLOW),
|
DRM_RENDER_ALLOW),
|
||||||
DRM_IOCTL_DEF_DRV(TEGRA_CHANNEL_CLOSE, tegra_drm_ioctl_channel_close,
|
DRM_IOCTL_DEF_DRV(TEGRA_CHANNEL_CLOSE, tegra_drm_ioctl_channel_close,
|
||||||
@@ -745,7 +743,6 @@ static const struct drm_ioctl_desc tegra_drm_ioctls[] = {
|
|||||||
DRM_RENDER_ALLOW),
|
DRM_RENDER_ALLOW),
|
||||||
DRM_IOCTL_DEF_DRV(TEGRA_GEM_MMAP, tegra_drm_ioctl_gem_mmap,
|
DRM_IOCTL_DEF_DRV(TEGRA_GEM_MMAP, tegra_drm_ioctl_gem_mmap,
|
||||||
DRM_RENDER_ALLOW),
|
DRM_RENDER_ALLOW),
|
||||||
#ifdef CONFIG_DRM_TEGRA_STAGING
|
|
||||||
|
|
||||||
DRM_IOCTL_DEF_DRV(TEGRA_GEM_CREATE_LEGACY, tegra_gem_create, DRM_RENDER_ALLOW),
|
DRM_IOCTL_DEF_DRV(TEGRA_GEM_CREATE_LEGACY, tegra_gem_create, DRM_RENDER_ALLOW),
|
||||||
DRM_IOCTL_DEF_DRV(TEGRA_GEM_MMAP_LEGACY, tegra_gem_mmap, DRM_RENDER_ALLOW),
|
DRM_IOCTL_DEF_DRV(TEGRA_GEM_MMAP_LEGACY, tegra_gem_mmap, DRM_RENDER_ALLOW),
|
||||||
|
|||||||
@@ -12,6 +12,7 @@ struct tegra_drm_firewall {
|
|||||||
u32 *data;
|
u32 *data;
|
||||||
u32 pos;
|
u32 pos;
|
||||||
u32 end;
|
u32 end;
|
||||||
|
u32 class;
|
||||||
};
|
};
|
||||||
|
|
||||||
static int fw_next(struct tegra_drm_firewall *fw, u32 *word)
|
static int fw_next(struct tegra_drm_firewall *fw, u32 *word)
|
||||||
@@ -51,8 +52,8 @@ static int fw_check_reg(struct tegra_drm_firewall *fw, u32 offset)
|
|||||||
if (!fw->client->ops->is_addr_reg)
|
if (!fw->client->ops->is_addr_reg)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
is_addr = fw->client->ops->is_addr_reg(
|
is_addr = fw->client->ops->is_addr_reg(fw->client->base.dev, fw->class,
|
||||||
fw->client->base.dev, fw->client->base.class, offset);
|
offset);
|
||||||
|
|
||||||
if (!is_addr)
|
if (!is_addr)
|
||||||
return 0;
|
return 0;
|
||||||
@@ -97,14 +98,25 @@ static int fw_check_regs_imm(struct tegra_drm_firewall *fw, u32 offset)
|
|||||||
{
|
{
|
||||||
bool is_addr;
|
bool is_addr;
|
||||||
|
|
||||||
is_addr = fw->client->ops->is_addr_reg(fw->client->base.dev,
|
is_addr = fw->client->ops->is_addr_reg(fw->client->base.dev, fw->class,
|
||||||
fw->client->base.class, offset);
|
offset);
|
||||||
if (is_addr)
|
if (is_addr)
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int fw_check_class(struct tegra_drm_firewall *fw, u32 class)
|
||||||
|
{
|
||||||
|
if (!fw->client->ops->is_valid_class)
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
if (!fw->client->ops->is_valid_class(class))
|
||||||
|
return -EINVAL;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
enum {
|
enum {
|
||||||
HOST1X_OPCODE_SETCLASS = 0x00,
|
HOST1X_OPCODE_SETCLASS = 0x00,
|
||||||
HOST1X_OPCODE_INCR = 0x01,
|
HOST1X_OPCODE_INCR = 0x01,
|
||||||
@@ -124,7 +136,8 @@ enum {
|
|||||||
};
|
};
|
||||||
|
|
||||||
int tegra_drm_fw_validate(struct tegra_drm_client *client, u32 *data, u32 start,
|
int tegra_drm_fw_validate(struct tegra_drm_client *client, u32 *data, u32 start,
|
||||||
u32 words, struct tegra_drm_submit_data *submit)
|
u32 words, struct tegra_drm_submit_data *submit,
|
||||||
|
u32 *job_class)
|
||||||
{
|
{
|
||||||
struct tegra_drm_firewall fw = {
|
struct tegra_drm_firewall fw = {
|
||||||
.submit = submit,
|
.submit = submit,
|
||||||
@@ -132,13 +145,14 @@ int tegra_drm_fw_validate(struct tegra_drm_client *client, u32 *data, u32 start,
|
|||||||
.data = data,
|
.data = data,
|
||||||
.pos = start,
|
.pos = start,
|
||||||
.end = start+words,
|
.end = start+words,
|
||||||
|
.class = *job_class,
|
||||||
};
|
};
|
||||||
bool payload_valid = false;
|
bool payload_valid = false;
|
||||||
u32 payload;
|
u32 payload;
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
while (fw.pos != fw.end) {
|
while (fw.pos != fw.end) {
|
||||||
u32 word, opcode, offset, count, mask;
|
u32 word, opcode, offset, count, mask, class;
|
||||||
|
|
||||||
err = fw_next(&fw, &word);
|
err = fw_next(&fw, &word);
|
||||||
if (err)
|
if (err)
|
||||||
@@ -147,6 +161,16 @@ int tegra_drm_fw_validate(struct tegra_drm_client *client, u32 *data, u32 start,
|
|||||||
opcode = (word & 0xf0000000) >> 28;
|
opcode = (word & 0xf0000000) >> 28;
|
||||||
|
|
||||||
switch (opcode) {
|
switch (opcode) {
|
||||||
|
case HOST1X_OPCODE_SETCLASS:
|
||||||
|
offset = word >> 16 & 0xfff;
|
||||||
|
mask = word & 0x3f;
|
||||||
|
class = (word >> 6) & 0x3ff;
|
||||||
|
err = fw_check_class(&fw, class);
|
||||||
|
fw.class = class;
|
||||||
|
*job_class = class;
|
||||||
|
if (!err)
|
||||||
|
err = fw_check_regs_mask(&fw, offset, mask);
|
||||||
|
break;
|
||||||
case HOST1X_OPCODE_INCR:
|
case HOST1X_OPCODE_INCR:
|
||||||
offset = (word >> 16) & 0xfff;
|
offset = (word >> 16) & 0xfff;
|
||||||
count = word & 0xffff;
|
count = word & 0xffff;
|
||||||
|
|||||||
@@ -44,6 +44,9 @@ static void *alloc_copy_user_array(void __user *from, size_t count, size_t size)
|
|||||||
if (check_mul_overflow(count, size, ©_len))
|
if (check_mul_overflow(count, size, ©_len))
|
||||||
return ERR_PTR(-EINVAL);
|
return ERR_PTR(-EINVAL);
|
||||||
|
|
||||||
|
if (copy_len > 0x4000)
|
||||||
|
return ERR_PTR(-E2BIG);
|
||||||
|
|
||||||
data = kvmalloc(copy_len, GFP_KERNEL);
|
data = kvmalloc(copy_len, GFP_KERNEL);
|
||||||
if (!data)
|
if (!data)
|
||||||
return ERR_PTR(-ENOMEM);
|
return ERR_PTR(-ENOMEM);
|
||||||
@@ -110,8 +113,10 @@ static int submit_write_reloc(struct gather_bo *bo,
|
|||||||
dma_addr_t iova = mapping->iova + buf->reloc.target_offset;
|
dma_addr_t iova = mapping->iova + buf->reloc.target_offset;
|
||||||
u32 written_ptr = (u32)(iova >> buf->reloc.shift);
|
u32 written_ptr = (u32)(iova >> buf->reloc.shift);
|
||||||
|
|
||||||
|
#ifdef CONFIG_ARM64
|
||||||
if (buf->flags & DRM_TEGRA_SUBMIT_BUF_RELOC_BLOCKLINEAR)
|
if (buf->flags & DRM_TEGRA_SUBMIT_BUF_RELOC_BLOCKLINEAR)
|
||||||
written_ptr |= BIT(39);
|
written_ptr |= BIT(39);
|
||||||
|
#endif
|
||||||
|
|
||||||
if (buf->reloc.gather_offset_words >= bo->gather_data_words)
|
if (buf->reloc.gather_offset_words >= bo->gather_data_words)
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
@@ -219,7 +224,8 @@ static int submit_job_add_gather(struct host1x_job *job,
|
|||||||
struct tegra_drm_channel_ctx *ctx,
|
struct tegra_drm_channel_ctx *ctx,
|
||||||
struct drm_tegra_submit_cmd_gather_uptr *cmd,
|
struct drm_tegra_submit_cmd_gather_uptr *cmd,
|
||||||
struct gather_bo *bo, u32 *offset,
|
struct gather_bo *bo, u32 *offset,
|
||||||
struct tegra_drm_submit_data *job_data)
|
struct tegra_drm_submit_data *job_data,
|
||||||
|
u32 *class)
|
||||||
{
|
{
|
||||||
u32 next_offset;
|
u32 next_offset;
|
||||||
|
|
||||||
@@ -237,7 +243,7 @@ static int submit_job_add_gather(struct host1x_job *job,
|
|||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
if (tegra_drm_fw_validate(ctx->client, bo->gather_data, *offset,
|
if (tegra_drm_fw_validate(ctx->client, bo->gather_data, *offset,
|
||||||
cmd->words, job_data))
|
cmd->words, job_data, class))
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
|
|
||||||
host1x_job_add_gather(job, &bo->base, cmd->words, *offset * 4);
|
host1x_job_add_gather(job, &bo->base, cmd->words, *offset * 4);
|
||||||
@@ -254,10 +260,13 @@ static int submit_create_job(struct drm_device *drm, struct host1x_job **pjob,
|
|||||||
struct tegra_drm_submit_data *job_data)
|
struct tegra_drm_submit_data *job_data)
|
||||||
{
|
{
|
||||||
struct drm_tegra_submit_cmd *cmds;
|
struct drm_tegra_submit_cmd *cmds;
|
||||||
u32 i, gather_offset = 0;
|
u32 i, gather_offset = 0, class;
|
||||||
struct host1x_job *job;
|
struct host1x_job *job;
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
|
/* Set initial class for firewall. */
|
||||||
|
class = ctx->client->base.class;
|
||||||
|
|
||||||
cmds = alloc_copy_user_array(u64_to_user_ptr(args->cmds_ptr),
|
cmds = alloc_copy_user_array(u64_to_user_ptr(args->cmds_ptr),
|
||||||
args->num_cmds, sizeof(*cmds));
|
args->num_cmds, sizeof(*cmds));
|
||||||
if (IS_ERR(cmds))
|
if (IS_ERR(cmds))
|
||||||
@@ -283,7 +292,7 @@ static int submit_create_job(struct drm_device *drm, struct host1x_job **pjob,
|
|||||||
if (cmd->type == DRM_TEGRA_SUBMIT_CMD_GATHER_UPTR) {
|
if (cmd->type == DRM_TEGRA_SUBMIT_CMD_GATHER_UPTR) {
|
||||||
err = submit_job_add_gather(job, ctx, &cmd->gather_uptr,
|
err = submit_job_add_gather(job, ctx, &cmd->gather_uptr,
|
||||||
bo, &gather_offset,
|
bo, &gather_offset,
|
||||||
job_data);
|
job_data, &class);
|
||||||
if (err)
|
if (err)
|
||||||
goto free_job;
|
goto free_job;
|
||||||
} else if (cmd->type == DRM_TEGRA_SUBMIT_CMD_WAIT_SYNCPT) {
|
} else if (cmd->type == DRM_TEGRA_SUBMIT_CMD_WAIT_SYNCPT) {
|
||||||
|
|||||||
@@ -15,6 +15,7 @@ struct tegra_drm_submit_data {
|
|||||||
};
|
};
|
||||||
|
|
||||||
int tegra_drm_fw_validate(struct tegra_drm_client *client, u32 *data, u32 start,
|
int tegra_drm_fw_validate(struct tegra_drm_client *client, u32 *data, u32 start,
|
||||||
u32 words, struct tegra_drm_submit_data *submit);
|
u32 words, struct tegra_drm_submit_data *submit,
|
||||||
|
u32 *job_class);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|||||||
@@ -215,9 +215,10 @@ int tegra_drm_ioctl_channel_map(struct drm_device *drm, void *data,
|
|||||||
|
|
||||||
/* TODO only map the requested part */
|
/* TODO only map the requested part */
|
||||||
mapping->iova = sg_dma_address(mapping->sgt->sgl);
|
mapping->iova = sg_dma_address(mapping->sgt->sgl);
|
||||||
mapping->iova_end = mapping->iova + gem->size;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
mapping->iova_end = mapping->iova + gem->size;
|
||||||
|
|
||||||
mutex_unlock(&fpriv->lock);
|
mutex_unlock(&fpriv->lock);
|
||||||
|
|
||||||
err = xa_alloc(&ctx->mappings, &mapping_id, mapping,
|
err = xa_alloc(&ctx->mappings, &mapping_id, mapping,
|
||||||
|
|||||||
Reference in New Issue
Block a user