Experiment compatibility fix: OBU_SIZING and OBU_NO_IVF.
Make OBU_SIZING and OBU_NO_IVF work when combined. Correct the
support for OBUs with LEB128 size fields in obudec.c.
In addition: explicitly disallow pipe input of Annex B OBU without
framing (this implementation requires seeking backwards to function).
Change-Id: I331371f51cae7752a9b1fc091174fe32816ea027
diff --git a/obudec.c b/obudec.c
index a0cb85e..d3443c4 100644
--- a/obudec.c
+++ b/obudec.c
@@ -19,11 +19,67 @@
#include "av1/common/common.h"
#define OBU_HEADER_SIZE_BYTES 1
-#if CONFIG_SCALABILITY
#define OBU_HEADER_EXTENSION_SIZE_BYTES 1
+
+#if CONFIG_OBU_SIZING
+// Unsigned LEB128 OBU length field has maximum size of 8 bytes.
+#define OBU_MAX_LENGTH_FIELD_SIZE 8
+#define OBU_MAX_HEADER_SIZE \
+ (OBU_HEADER_SIZE_BYTES + OBU_HEADER_EXTENSION_SIZE_BYTES + \
+ OBU_MAX_LENGTH_FIELD_SIZE)
+#else
+#define OBU_MAX_LENGTH_FIELD_SIZE PRE_OBU_SIZE_BYTES
+#define OBU_MAX_HEADER_SIZE \
+ (OBU_HEADER_SIZE_BYTES + OBU_HEADER_EXTENSION_SIZE_BYTES + PRE_OBU_SIZE_BYTES)
#endif
#if CONFIG_OBU_NO_IVF
+int read_obu_type(uint8_t obu_header, OBU_TYPE *obu_type) {
+ if (!obu_type) return -1;
+ const int obu_type_value = (obu_header >> 3) & 0xF;
+ switch (obu_type_value) {
+ case OBU_SEQUENCE_HEADER: *obu_type = OBU_SEQUENCE_HEADER; break;
+ case OBU_TEMPORAL_DELIMITER: *obu_type = OBU_TEMPORAL_DELIMITER; break;
+ case OBU_FRAME_HEADER: *obu_type = OBU_FRAME_HEADER; break;
+ case OBU_TILE_GROUP: *obu_type = OBU_TILE_GROUP; break;
+ case OBU_METADATA: *obu_type = OBU_METADATA; break;
+ case OBU_PADDING: *obu_type = OBU_PADDING; break;
+ default: return -1;
+ }
+ return 0;
+}
+
+// Reads OBU size from infile and returns 0 upon success. Returns obu_size via
+// output pointer obu_size. Returns -1 when reading or parsing fails. Always
+// returns FILE pointer to position at time of call. Returns 0 and sets obu_size
+// to 0 when end of file is reached.
+int read_obu_size(FILE *infile, uint64_t *obu_size, size_t *length_field_size) {
+ if (!infile || !obu_size) return 1;
+
+ uint8_t read_buffer[OBU_MAX_LENGTH_FIELD_SIZE] = { 0 };
+ size_t bytes_read = fread(read_buffer, 1, OBU_MAX_LENGTH_FIELD_SIZE, infile);
+ *obu_size = 0;
+
+ if (bytes_read == 0) {
+ return 0;
+ }
+
+ const int seek_pos = (int)bytes_read;
+ if (seek_pos != 0 && fseek(infile, -seek_pos, SEEK_CUR) != 0) return 1;
+
+#if CONFIG_OBU_SIZING
+ if (aom_uleb_decode(read_buffer, bytes_read, obu_size, length_field_size) !=
+ 0) {
+ return 1;
+ }
+#else
+ if (length_field_size) *length_field_size = PRE_OBU_SIZE_BYTES;
+ *obu_size = mem_get_le32(read_buffer);
+#endif
+
+ return 0;
+}
+
int obu_read_temporal_unit(FILE *infile, uint8_t **buffer, size_t *bytes_read,
#if CONFIG_SCALABILITY
size_t *buffer_size, int last_layer_id) {
@@ -31,8 +87,6 @@
size_t *buffer_size) {
#endif
size_t ret;
- const size_t obu_length_header_size =
- PRE_OBU_SIZE_BYTES + OBU_HEADER_SIZE_BYTES;
uint64_t obu_size = 0;
uint8_t *data = NULL;
@@ -43,109 +97,116 @@
*buffer_size = 0;
*bytes_read = 0;
while (1) {
- // augment the buffer to just contain the next size field
- // and the first byte of the header
- *buffer = realloc(*buffer, (*buffer_size) + obu_length_header_size);
- data = *buffer + (*buffer_size);
- *buffer_size += obu_length_header_size;
- ret = fread(data, 1, obu_length_header_size, infile);
- if (ret == 0) {
- // fprintf(stderr, "Found end of stream, ending temporal unit\n");
- break;
- }
- if (ret != obu_length_header_size) {
- warn("Failed to read OBU Header\n");
+ size_t length_field_size = 0;
+ ret = read_obu_size(infile, &obu_size, &length_field_size);
+ if (ret != 0) {
+ warn("Failed to read OBU size.\n");
return 1;
}
- *bytes_read += obu_length_header_size;
+ if (ret == 0 && obu_size == 0) {
+ fprintf(stderr, "Found end of stream, ending temporal unit\n");
+ break;
+ }
- if (((data[PRE_OBU_SIZE_BYTES] >> 3) & 0xF) == OBU_TEMPORAL_DELIMITER) {
+ obu_size += length_field_size;
+
+ // Expand the buffer to contain the full OBU and its length.
+ const uint8_t *old_buffer = *buffer;
+ *buffer = (uint8_t *)realloc(*buffer, *buffer_size + obu_size);
+ if (!*buffer) {
+ free((void *)old_buffer);
+ warn("OBU buffer alloc failed.\n");
+ return 1;
+ }
+
+ data = *buffer + (*buffer_size);
+ *buffer_size += obu_size;
+ ret = fread(data, 1, obu_size, infile);
+
+ if (ret != obu_size) {
+ warn("Failed to read OBU.\n");
+ return 1;
+ }
+ *bytes_read += obu_size;
+
+ OBU_TYPE obu_type;
+ if (read_obu_type(data[length_field_size], &obu_type) != 0) {
+ warn("Invalid OBU type.\n");
+ return 1;
+ }
+
+ if (obu_type == OBU_TEMPORAL_DELIMITER) {
// Stop when a temporal delimiter is found
// fprintf(stderr, "Found temporal delimiter, ending temporal unit\n");
// prevent decoder to start decoding another frame from this buffer
- *bytes_read -= obu_length_header_size;
+ *bytes_read -= obu_size;
break;
}
#if CONFIG_SCALABILITY
// break if obu_extension_flag is found and enhancement_id change
- if ((data[PRE_OBU_SIZE_BYTES] & 0x1)) {
- uint8_t obu_extension_header;
- const int total_obu_header_size =
- (int)obu_length_header_size + OBU_HEADER_EXTENSION_SIZE_BYTES;
- int curr_layer_id;
- ret = fread(&obu_extension_header, 1, OBU_HEADER_EXTENSION_SIZE_BYTES,
- infile);
- if (ret != OBU_HEADER_EXTENSION_SIZE_BYTES) {
- warn("Failed to read OBU Header Extension\n");
- return 1;
- }
- curr_layer_id = (obu_extension_header >> 3) & 0x3;
+ if (data[length_field_size] & 0x1) {
+ const uint8_t obu_extension_header =
+ data[length_field_size + OBU_HEADER_SIZE_BYTES];
+ const int curr_layer_id = (obu_extension_header >> 3) & 0x3;
if (curr_layer_id && (curr_layer_id > last_layer_id)) {
// new enhancement layer
- *bytes_read -= obu_length_header_size;
- fseek(infile, -total_obu_header_size, SEEK_CUR);
+ *bytes_read -= obu_size;
+ const int i_obu_size = (int)obu_size;
+ fseek(infile, -i_obu_size, SEEK_CUR);
break;
- } else {
- fseek(infile, -OBU_HEADER_EXTENSION_SIZE_BYTES, SEEK_CUR);
}
}
#endif
-
-// otherwise, read the OBU payload into memory
-#if CONFIG_OBU_SIZING
- aom_uleb_decode(data, PRE_OBU_SIZE_BYTES, &obu_size, NULL);
-#else
- obu_size = mem_get_le32(data);
-#endif // CONFIG_OBU_SIZING
-
- // fprintf(stderr, "Found OBU of type %d and size %d\n",
- // ((data[PRE_OBU_SIZE_BYTES] >> 3) & 0xF), obu_size);
- obu_size--; // removing the byte of the header already read
- if (obu_size) {
- *buffer = realloc(*buffer, (*buffer_size) + obu_size);
- data = *buffer + (*buffer_size);
- *buffer_size += obu_size;
- ret = fread(data, 1, obu_size, infile);
- if (ret != obu_size) {
- warn("Failed to read OBU Payload\n");
- return 1;
- }
- *bytes_read += obu_size;
- }
}
+
return 0;
}
int file_is_obu(struct AvxInputContext *input_ctx) {
- uint8_t obutd[PRE_OBU_SIZE_BYTES + OBU_HEADER_SIZE_BYTES];
- uint64_t size;
+ uint8_t obutd[OBU_MAX_HEADER_SIZE] = { 0 };
+ uint64_t size = 0;
+
+ // Parsing of Annex B OBU streams via pipe without framing not supported. This
+ // implementation requires seeking backwards in the input stream. Tell caller
+ // that this input cannot be processed.
+ if (!input_ctx->filename || strcmp(input_ctx->filename, "-") == 0) return 0;
// Reading the first OBU TD to enable TU end detection at TD start.
- const size_t obu_length_header_size =
- PRE_OBU_SIZE_BYTES + OBU_HEADER_SIZE_BYTES;
- const size_t ret = fread(obutd, 1, obu_length_header_size, input_ctx->file);
- if (ret != obu_length_header_size) {
- warn("Failed to read OBU Header\n");
+ const size_t ret = fread(obutd, 1, OBU_MAX_HEADER_SIZE, input_ctx->file);
+ if (ret != OBU_MAX_HEADER_SIZE) {
+ warn("Failed to read OBU Header, not enough data to process header.\n");
return 0;
}
#if CONFIG_OBU_SIZING
- aom_uleb_decode(obutd, PRE_OBU_SIZE_BYTES, &size, NULL);
+ if (aom_uleb_decode(obutd, OBU_MAX_HEADER_SIZE, &size, NULL) != 0) {
+ warn("OBU size parse failed.\n");
+ return 0;
+ }
+ const size_t obu_header_offset = aom_uleb_size_in_bytes(size);
#else
+ const size_t obu_header_offset = PRE_OBU_SIZE_BYTES;
size = mem_get_le32(obutd);
#endif // CONFIG_OBU_SIZING
- if (size != 1) {
- warn("Expected first OBU size to be 1, got %d", size);
- return 0;
- }
- if (((obutd[PRE_OBU_SIZE_BYTES] >> 3) & 0xF) != OBU_TEMPORAL_DELIMITER) {
- warn("Expected OBU TD at file start, got %d\n", obutd[PRE_OBU_SIZE_BYTES]);
- return 0;
- }
- // fprintf(stderr, "Starting to parse OBU stream\n");
+ fseek(input_ctx->file, obu_header_offset + OBU_HEADER_SIZE_BYTES, SEEK_SET);
+ if (size != 1) {
+ warn("Expected first OBU size to be 1, got %d\n", size);
+ return 0;
+ }
+ OBU_TYPE obu_type;
+ if (read_obu_type(obutd[obu_header_offset], &obu_type) != 0) {
+ warn("Invalid OBU type found while probing for OBU_TEMPORAL_DELIMITER.\n");
+ return 0;
+ }
+ if (obu_type != OBU_TEMPORAL_DELIMITER) {
+ warn("Expected OBU TD at file start, got %d\n", obutd[obu_header_offset]);
+ return 0;
+ }
+
+ // fprintf(stderr, "Starting to parse OBU stream\n");
return 1;
}