Add obu_sizing experiment.

Writes PRE_OBU_SIZE_BYTES (currently 4) bytes padded unsigned LEB128
encoded integers in OBU size fields when enabled:

$ cmake path/to/aom -DCONFIG_OBU=1 -DCONFIG_OBU_SIZING=1 && cmake --build .

Requires CONFIG_OBU.

BUG=aomedia:1125

Change-Id: I4d184ef0c8587d24e9c8c3e63237ea5003386c6a
diff --git a/av1/av1_cx_iface.c b/av1/av1_cx_iface.c
index 385111d..2eb0344 100644
--- a/av1/av1_cx_iface.c
+++ b/av1/av1_cx_iface.c
@@ -1400,7 +1400,7 @@
 
 #if CONFIG_OBU
       // move data PRE_OBU_SIZE_BYTES + 1 bytes and insert OBU_TD preceded by
-      // optional 4 byte size
+      // 4 byte size
       uint32_t obu_size = 1;
       if (ctx->pending_cx_data) {
         const size_t index_sz = PRE_OBU_SIZE_BYTES + 1;
@@ -1411,9 +1411,17 @@
           OBU_TEMPORAL_DELIMITER, 0,
           (uint8_t *)(ctx->pending_cx_data + PRE_OBU_SIZE_BYTES));
       obu_size += write_temporal_delimiter_obu();
+#if CONFIG_OBU_SIZING
+      // OBUs are preceded by an unsigned leb128 coded unsigned integer padded
+      // to PRE_OBU_SIZE_BYTES bytes.
+      if (write_uleb_obu_size(obu_size, ctx->pending_cx_data) != AOM_CODEC_OK)
+        return AOM_CODEC_ERROR;
+#else
       mem_put_le32(ctx->pending_cx_data, obu_size);
+#endif  // CONFIG_OBU_SIZING
+
       pkt.data.frame.sz += (obu_size + PRE_OBU_SIZE_BYTES);
-#endif
+#endif  // CONFIG_OBU
 
       pkt.data.frame.pts = ticks_to_timebase_units(timebase, dst_time_stamp);
       pkt.data.frame.flags = get_frame_pkt_flags(cpi, lib_flags);
diff --git a/av1/decoder/obu.c b/av1/decoder/obu.c
index 2586ed5..ef183b9 100644
--- a/av1/decoder/obu.c
+++ b/av1/decoder/obu.c
@@ -175,10 +175,20 @@
     struct aom_read_bit_buffer rb;
     size_t obu_header_size, obu_payload_size = 0;
     av1_init_read_bit_buffer(pbi, &rb, data + PRE_OBU_SIZE_BYTES, data_end);
+
+#if CONFIG_OBU_SIZING
+    // OBUs are preceded by an unsigned leb128 coded unsigned integer padded to
+    // PRE_OBU_SIZE_BYTES bytes.
+    uint32_t u_obu_size = 0;
+    aom_uleb_decode(data, PRE_OBU_SIZE_BYTES, &u_obu_size);
+    const size_t obu_size = (size_t)u_obu_size;
+#else
     // every obu is preceded by PRE_OBU_SIZE_BYTES-byte size of obu (obu header
     // + payload size)
     // The obu size is only needed for tile group OBUs
     const size_t obu_size = mem_get_le32(data);
+#endif  // CONFIG_OBU_SIZING
+
     const OBU_TYPE obu_type = read_obu_header(&rb, &obu_header_size);
     data += (PRE_OBU_SIZE_BYTES + obu_header_size);
     if (data_end < data) {
diff --git a/av1/encoder/bitstream.c b/av1/encoder/bitstream.c
index 8fa0ac5..1f2a33c 100644
--- a/av1/encoder/bitstream.c
+++ b/av1/encoder/bitstream.c
@@ -4367,6 +4367,22 @@
   return size;
 }
 
+#if CONFIG_OBU_SIZING
+int write_uleb_obu_size(uint32_t obu_size, uint8_t *dest) {
+  size_t coded_obu_size = 0;
+
+  // Encode an unsigned leb128 coded unsigned integer padded to
+  // PRE_OBU_SIZE_BYTES bytes.
+  if (aom_uleb_encode_fixed_size(obu_size, PRE_OBU_SIZE_BYTES,
+                                 PRE_OBU_SIZE_BYTES, dest, &coded_obu_size) ||
+      coded_obu_size != PRE_OBU_SIZE_BYTES) {
+    return AOM_CODEC_ERROR;
+  }
+
+  return AOM_CODEC_OK;
+}
+#endif  // CONFIG_OBU_SIZING
+
 static uint32_t write_sequence_header_obu(AV1_COMP *cpi, uint8_t *const dst) {
   AV1_COMMON *const cm = &cpi->common;
   struct aom_write_bit_buffer wb = { dst, 0 };
@@ -4670,8 +4686,13 @@
           // size of this tile
           mem_put_le32(buf->data, tile_size);
         } else {
-          // write current tile group size
-          mem_put_le32(data, curr_tg_data_size);
+// write current tile group size
+#if CONFIG_OBU_SIZING
+          if (write_uleb_obu_size(curr_tg_data_size, data) != AOM_CODEC_OK)
+            assert(0);
+#else
+        mem_put_le32(data, curr_tg_data_size);
+#endif  // CONFIG_OBU_SIZING
         }
 
         total_size += tile_size;
@@ -4683,7 +4704,7 @@
   return (uint32_t)total_size;
 }
 
-#endif
+#endif  // CONFIG_OBU
 
 int av1_pack_bitstream(AV1_COMP *const cpi, uint8_t *dst, size_t *size) {
   uint8_t *data = dst;
@@ -4695,7 +4716,7 @@
   uint32_t obu_size;
   uint8_t *frame_header_location;
   uint32_t frame_header_size;
-#endif
+#endif  // CONFIG_OBU
   (void)cm;
 #if CONFIG_BITSTREAM_DEBUG
   bitstream_queue_reset_write();
@@ -4710,7 +4731,14 @@
         write_obu_header(OBU_SEQUENCE_HEADER, 0, data + PRE_OBU_SIZE_BYTES);
     obu_size +=
         write_sequence_header_obu(cpi, data + PRE_OBU_SIZE_BYTES + obu_size);
+
+#if CONFIG_OBU_SIZING
+    if (write_uleb_obu_size(obu_size, data) != AOM_CODEC_OK)
+      return AOM_CODEC_ERROR;
+#else
     mem_put_le32(data, obu_size);
+#endif  // CONFIG_OBU_SIZING
+
     data += obu_size + PRE_OBU_SIZE_BYTES;
   }
 
@@ -4728,7 +4756,14 @@
 #endif
                              data + PRE_OBU_SIZE_BYTES + obu_size);
   obu_size += frame_header_size;
+
+#if CONFIG_OBU_SIZING
+  if (write_uleb_obu_size(obu_size, data) != AOM_CODEC_OK)
+    return AOM_CODEC_ERROR;
+#else
   mem_put_le32(data, obu_size);
+#endif  // CONFIG_OBU_SIZING
+
   data += obu_size + PRE_OBU_SIZE_BYTES;
 
   if (cm->show_existing_frame) {
diff --git a/av1/encoder/bitstream.h b/av1/encoder/bitstream.h
index 51ea9b6..c7fe509 100644
--- a/av1/encoder/bitstream.h
+++ b/av1/encoder/bitstream.h
@@ -27,7 +27,11 @@
 #if CONFIG_OBU
 uint32_t write_obu_header(OBU_TYPE obu_type, int obu_extension,
                           uint8_t *const dst);
-#endif
+
+#if CONFIG_OBU_SIZING
+int write_uleb_obu_size(uint32_t obu_size, uint8_t *dest);
+#endif  // CONFIG_OBU_SIZING
+#endif  // CONFIG_OBU
 
 int av1_pack_bitstream(AV1_COMP *const cpi, uint8_t *dest, size_t *size);
 
diff --git a/build/cmake/aom_config_defaults.cmake b/build/cmake/aom_config_defaults.cmake
index 1ae86b9..2efe62f 100644
--- a/build/cmake/aom_config_defaults.cmake
+++ b/build/cmake/aom_config_defaults.cmake
@@ -148,6 +148,7 @@
 set(CONFIG_NO_FRAME_CONTEXT_SIGNALING 0 CACHE NUMBER "AV1 experiment flag.")
 set(CONFIG_OBU 1 CACHE NUMBER "AV1 experiment flag.")
 set(CONFIG_OBU_NO_IVF 0 CACHE NUMBER "AV1 experiment flag.")
+set(CONFIG_OBU_SIZING 0 CACHE NUMBER "AV1 experiment flag.")
 set(CONFIG_OPT_REF_MV 0 CACHE NUMBER "AV1 experiment flag.")
 set(CONFIG_PALETTE_THROUGHPUT 1 CACHE NUMBER "AV1 experiment flag.")
 set(CONFIG_PARALLEL_DEBLOCKING 1 CACHE NUMBER "AV1 experiment flag.")
diff --git a/obudec.c b/obudec.c
index 9cc907a..8f1ca00 100644
--- a/obudec.c
+++ b/obudec.c
@@ -60,8 +60,13 @@
       break;
     }
 
-    // otherwise, read the OBU payload into memory
+// otherwise, read the OBU payload into memory
+#if CONFIG_OBU_SIZING
+    aom_uleb_decode(data, PRE_OBU_SIZE_BYTES, &obu_size);
+#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
@@ -82,7 +87,7 @@
 
 int file_is_obu(struct AvxInputContext *input_ctx) {
   uint8_t obutd[PRE_OBU_SIZE_BYTES + OBU_HEADER_SIZE_BYTES];
-  int size;
+  uint32_t size;
 
 #if !CONFIG_OBU
   warn("obudec.c requires CONFIG_OBU");
@@ -91,7 +96,13 @@
 
   // Reading the first OBU TD to enable TU end detection at TD start.
   fread(obutd, 1, PRE_OBU_SIZE_BYTES + OBU_HEADER_SIZE_BYTES, input_ctx->file);
+
+#if CONFIG_OBU_SIZING
+  aom_uleb_decode(obutd, PRE_OBU_SIZE_BYTES, &size);
+#else
   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;
diff --git a/tools/obu_parser.cc b/tools/obu_parser.cc
index 6c5f23b..d63a484 100644
--- a/tools/obu_parser.cc
+++ b/tools/obu_parser.cc
@@ -21,6 +21,7 @@
 #error "obu_parser.cc requires CONFIG_OBU"
 #endif
 
+#include "aom/aom_integer.h"
 #include "aom_ports/mem_ops.h"
 #include "tools/obu_parser.h"
 
@@ -170,7 +171,14 @@
       return false;
     }
 
+#if CONFIG_OBU_SIZING
+    uint32_t obu_size = 0;
+    aom_uleb_decode(data + consumed, kObuLengthFieldSizeBytes, &obu_size);
+    const int current_obu_length = static_cast<int>(obu_size);
+#else
     const int current_obu_length = mem_get_le32(data + consumed);
+#endif  // CONFIG_OBU_SIZING
+
     if (current_obu_length > remaining) {
       fprintf(stderr,
               "OBU parsing failed at offset %d with bad length of %d "