mirror of
https://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-01-18 02:46:06 +00:00
d2d79d29bb
Implement ioctls for the creation and destruction of contexts. Contexts are used for job submission and each is associated with a particular job type. Changes since v8: - Fixed one error path in pvr_stream_process_1() - Corrected license identifiers Changes since v5: - Fix context release in final error path in pvr_context_create() Changes since v3: - Use drm_dev_{enter,exit} Co-developed-by: Boris Brezillon <boris.brezillon@collabora.com> Signed-off-by: Boris Brezillon <boris.brezillon@collabora.com> Signed-off-by: Sarah Walker <sarah.walker@imgtec.com> Signed-off-by: Donald Robson <donald.robson@imgtec.com> Link: https://lore.kernel.org/r/ac474a1f7dda2582d290798e4837140a2989aa2a.1700668843.git.donald.robson@imgtec.com Signed-off-by: Maxime Ripard <mripard@kernel.org>
286 lines
7.7 KiB
C
286 lines
7.7 KiB
C
// SPDX-License-Identifier: GPL-2.0-only OR MIT
|
|
/* Copyright (c) 2023 Imagination Technologies Ltd. */
|
|
|
|
#include "pvr_device.h"
|
|
#include "pvr_rogue_fwif_stream.h"
|
|
#include "pvr_stream.h"
|
|
|
|
#include <linux/align.h>
|
|
#include <linux/slab.h>
|
|
#include <linux/types.h>
|
|
#include <uapi/drm/pvr_drm.h>
|
|
|
|
static __always_inline bool
|
|
stream_def_is_supported(struct pvr_device *pvr_dev, const struct pvr_stream_def *stream_def)
|
|
{
|
|
if (stream_def->feature == PVR_FEATURE_NONE)
|
|
return true;
|
|
|
|
if (!(stream_def->feature & PVR_FEATURE_NOT) &&
|
|
pvr_device_has_feature(pvr_dev, stream_def->feature)) {
|
|
return true;
|
|
}
|
|
|
|
if ((stream_def->feature & PVR_FEATURE_NOT) &&
|
|
!pvr_device_has_feature(pvr_dev, stream_def->feature & ~PVR_FEATURE_NOT)) {
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
static int
|
|
pvr_stream_get_data(u8 *stream, u32 *stream_offset, u32 stream_size, u32 data_size, u32 align_size,
|
|
void *dest)
|
|
{
|
|
*stream_offset = ALIGN(*stream_offset, align_size);
|
|
|
|
if ((*stream_offset + data_size) > stream_size)
|
|
return -EINVAL;
|
|
|
|
memcpy(dest, stream + *stream_offset, data_size);
|
|
|
|
(*stream_offset) += data_size;
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* pvr_stream_process_1() - Process a single stream and fill destination structure
|
|
* @pvr_dev: Device pointer.
|
|
* @stream_def: Stream definition.
|
|
* @nr_entries: Number of entries in &stream_def.
|
|
* @stream: Pointer to stream.
|
|
* @stream_offset: Starting offset within stream.
|
|
* @stream_size: Size of input stream, in bytes.
|
|
* @dest: Pointer to destination structure.
|
|
* @dest_size: Size of destination structure.
|
|
* @stream_offset_out: Pointer to variable to write updated stream offset to. May be NULL.
|
|
*
|
|
* Returns:
|
|
* * 0 on success, or
|
|
* * -%EINVAL on malformed stream.
|
|
*/
|
|
static int
|
|
pvr_stream_process_1(struct pvr_device *pvr_dev, const struct pvr_stream_def *stream_def,
|
|
u32 nr_entries, u8 *stream, u32 stream_offset, u32 stream_size,
|
|
u8 *dest, u32 dest_size, u32 *stream_offset_out)
|
|
{
|
|
int err = 0;
|
|
u32 i;
|
|
|
|
for (i = 0; i < nr_entries; i++) {
|
|
if (stream_def[i].offset >= dest_size) {
|
|
err = -EINVAL;
|
|
break;
|
|
}
|
|
|
|
if (!stream_def_is_supported(pvr_dev, &stream_def[i]))
|
|
continue;
|
|
|
|
switch (stream_def[i].size) {
|
|
case PVR_STREAM_SIZE_8:
|
|
err = pvr_stream_get_data(stream, &stream_offset, stream_size, sizeof(u8),
|
|
sizeof(u8), dest + stream_def[i].offset);
|
|
if (err)
|
|
return err;
|
|
break;
|
|
|
|
case PVR_STREAM_SIZE_16:
|
|
err = pvr_stream_get_data(stream, &stream_offset, stream_size, sizeof(u16),
|
|
sizeof(u16), dest + stream_def[i].offset);
|
|
if (err)
|
|
return err;
|
|
break;
|
|
|
|
case PVR_STREAM_SIZE_32:
|
|
err = pvr_stream_get_data(stream, &stream_offset, stream_size, sizeof(u32),
|
|
sizeof(u32), dest + stream_def[i].offset);
|
|
if (err)
|
|
return err;
|
|
break;
|
|
|
|
case PVR_STREAM_SIZE_64:
|
|
err = pvr_stream_get_data(stream, &stream_offset, stream_size, sizeof(u64),
|
|
sizeof(u64), dest + stream_def[i].offset);
|
|
if (err)
|
|
return err;
|
|
break;
|
|
|
|
case PVR_STREAM_SIZE_ARRAY:
|
|
err = pvr_stream_get_data(stream, &stream_offset, stream_size,
|
|
stream_def[i].array_size, sizeof(u64),
|
|
dest + stream_def[i].offset);
|
|
if (err)
|
|
return err;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (stream_offset_out)
|
|
*stream_offset_out = stream_offset;
|
|
|
|
return err;
|
|
}
|
|
|
|
static int
|
|
pvr_stream_process_ext_stream(struct pvr_device *pvr_dev,
|
|
const struct pvr_stream_cmd_defs *cmd_defs, void *ext_stream,
|
|
u32 stream_offset, u32 ext_stream_size, void *dest)
|
|
{
|
|
u32 musthave_masks[PVR_STREAM_EXTHDR_TYPE_MAX];
|
|
u32 ext_header;
|
|
int err = 0;
|
|
u32 i;
|
|
|
|
/* Copy "must have" mask from device. We clear this as we process the stream. */
|
|
memcpy(musthave_masks, pvr_dev->stream_musthave_quirks[cmd_defs->type],
|
|
sizeof(musthave_masks));
|
|
|
|
do {
|
|
const struct pvr_stream_ext_header *header;
|
|
u32 type;
|
|
u32 data;
|
|
|
|
err = pvr_stream_get_data(ext_stream, &stream_offset, ext_stream_size, sizeof(u32),
|
|
sizeof(ext_header), &ext_header);
|
|
if (err)
|
|
return err;
|
|
|
|
type = (ext_header & PVR_STREAM_EXTHDR_TYPE_MASK) >> PVR_STREAM_EXTHDR_TYPE_SHIFT;
|
|
data = ext_header & PVR_STREAM_EXTHDR_DATA_MASK;
|
|
|
|
if (type >= cmd_defs->ext_nr_headers)
|
|
return -EINVAL;
|
|
|
|
header = &cmd_defs->ext_headers[type];
|
|
if (data & ~header->valid_mask)
|
|
return -EINVAL;
|
|
|
|
musthave_masks[type] &= ~data;
|
|
|
|
for (i = 0; i < header->ext_streams_num; i++) {
|
|
const struct pvr_stream_ext_def *ext_def = &header->ext_streams[i];
|
|
|
|
if (!(ext_header & ext_def->header_mask))
|
|
continue;
|
|
|
|
if (!pvr_device_has_uapi_quirk(pvr_dev, ext_def->quirk))
|
|
return -EINVAL;
|
|
|
|
err = pvr_stream_process_1(pvr_dev, ext_def->stream, ext_def->stream_len,
|
|
ext_stream, stream_offset,
|
|
ext_stream_size, dest,
|
|
cmd_defs->dest_size, &stream_offset);
|
|
if (err)
|
|
return err;
|
|
}
|
|
} while (ext_header & PVR_STREAM_EXTHDR_CONTINUATION);
|
|
|
|
/*
|
|
* Verify that "must have" mask is now zero. If it isn't then one of the "must have" quirks
|
|
* for this command was not present.
|
|
*/
|
|
for (i = 0; i < cmd_defs->ext_nr_headers; i++) {
|
|
if (musthave_masks[i])
|
|
return -EINVAL;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* pvr_stream_process() - Build FW structure from stream
|
|
* @pvr_dev: Device pointer.
|
|
* @cmd_defs: Stream definition.
|
|
* @stream: Pointer to command stream.
|
|
* @stream_size: Size of command stream, in bytes.
|
|
* @dest_out: Pointer to destination buffer.
|
|
*
|
|
* Caller is responsible for freeing the output structure.
|
|
*
|
|
* Returns:
|
|
* * 0 on success,
|
|
* * -%ENOMEM on out of memory, or
|
|
* * -%EINVAL on malformed stream.
|
|
*/
|
|
int
|
|
pvr_stream_process(struct pvr_device *pvr_dev, const struct pvr_stream_cmd_defs *cmd_defs,
|
|
void *stream, u32 stream_size, void *dest_out)
|
|
{
|
|
u32 stream_offset = 0;
|
|
u32 main_stream_len;
|
|
u32 padding;
|
|
int err;
|
|
|
|
if (!stream || !stream_size)
|
|
return -EINVAL;
|
|
|
|
err = pvr_stream_get_data(stream, &stream_offset, stream_size, sizeof(u32),
|
|
sizeof(u32), &main_stream_len);
|
|
if (err)
|
|
return err;
|
|
|
|
/*
|
|
* u32 after stream length is padding to ensure u64 alignment, but may be used for expansion
|
|
* in the future. Verify it's zero.
|
|
*/
|
|
err = pvr_stream_get_data(stream, &stream_offset, stream_size, sizeof(u32),
|
|
sizeof(u32), &padding);
|
|
if (err)
|
|
return err;
|
|
|
|
if (main_stream_len < stream_offset || main_stream_len > stream_size || padding)
|
|
return -EINVAL;
|
|
|
|
err = pvr_stream_process_1(pvr_dev, cmd_defs->main_stream, cmd_defs->main_stream_len,
|
|
stream, stream_offset, main_stream_len, dest_out,
|
|
cmd_defs->dest_size, &stream_offset);
|
|
if (err)
|
|
return err;
|
|
|
|
if (stream_offset < stream_size) {
|
|
err = pvr_stream_process_ext_stream(pvr_dev, cmd_defs, stream, stream_offset,
|
|
stream_size, dest_out);
|
|
if (err)
|
|
return err;
|
|
} else {
|
|
u32 i;
|
|
|
|
/*
|
|
* If we don't have an extension stream then there must not be any "must have"
|
|
* quirks for this command.
|
|
*/
|
|
for (i = 0; i < cmd_defs->ext_nr_headers; i++) {
|
|
if (pvr_dev->stream_musthave_quirks[cmd_defs->type][i])
|
|
return -EINVAL;
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* pvr_stream_create_musthave_masks() - Create "must have" masks for streams based on current device
|
|
* quirks
|
|
* @pvr_dev: Device pointer.
|
|
*/
|
|
void
|
|
pvr_stream_create_musthave_masks(struct pvr_device *pvr_dev)
|
|
{
|
|
memset(pvr_dev->stream_musthave_quirks, 0, sizeof(pvr_dev->stream_musthave_quirks));
|
|
|
|
if (pvr_device_has_uapi_quirk(pvr_dev, 47217))
|
|
pvr_dev->stream_musthave_quirks[PVR_STREAM_TYPE_FRAG][0] |=
|
|
PVR_STREAM_EXTHDR_FRAG0_BRN47217;
|
|
|
|
if (pvr_device_has_uapi_quirk(pvr_dev, 49927)) {
|
|
pvr_dev->stream_musthave_quirks[PVR_STREAM_TYPE_GEOM][0] |=
|
|
PVR_STREAM_EXTHDR_GEOM0_BRN49927;
|
|
pvr_dev->stream_musthave_quirks[PVR_STREAM_TYPE_FRAG][0] |=
|
|
PVR_STREAM_EXTHDR_FRAG0_BRN49927;
|
|
pvr_dev->stream_musthave_quirks[PVR_STREAM_TYPE_COMPUTE][0] |=
|
|
PVR_STREAM_EXTHDR_COMPUTE0_BRN49927;
|
|
}
|
|
}
|