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.
*/ staticint
pvr_stream_process_1(struct pvr_device *pvr_dev, conststruct 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;
for (u32 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;
/* 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 { conststruct pvr_stream_ext_header *header;
u32 type;
u32 data;
/* * 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 (u32 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, conststruct 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;
/* * 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 (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 { /* * If we don't have an extension stream then there must not be any "must have" * quirks for this command.
*/ for (u32 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;
Die Informationen auf dieser Webseite wurden
nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit,
noch Qualität der bereit gestellten Informationen zugesichert.
Bemerkung:
Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.