MAX_TILE: Allow list of tile widths/heights

The aomenc API is modified to allow --tile_width and --tile_height to take a list of sizes.
For example, --tile_width=2,3 will give tiles of widths 2,3,2,3,... until the image width
is exhausted. --tile_width=2 will still work to gives tiles of width 2.

Change-Id: I2afa14d404557aa8b7341b20f7477590e03e0bdb
diff --git a/aom/aom_encoder.h b/aom/aom_encoder.h
index ff02847..c8fe996 100644
--- a/aom/aom_encoder.h
+++ b/aom/aom_encoder.h
@@ -619,6 +619,48 @@
    * implies a large-scale tile coding.
    */
   unsigned int large_scale_tile;
+
+  /*!\brief Number of explicit tile widths specified
+   *
+   * This value indicates the number of tile widths specified
+   * A value of 0 implies no tile widths are specified.
+   * Tile widths are given in the array tile_widths[]
+   */
+  int tile_width_count;
+
+  /*!\brief Number of explicit tile heights specified
+   *
+   * This value indicates the number of tile heights specified
+   * A value of 0 implies no tile heights are specified.
+   * Tile heights are given in the array tile_heights[]
+   */
+  int tile_height_count;
+
+/*!\brief Maximum number of tile widths in tile widths array
+ *
+ * This define gives the maximum number of elements in the tile_widths array.
+ */
+#define MAX_TILE_WIDTHS 64  // maximum tile width array length
+
+  /*!\brief Array of specified tile widths
+   *
+   * This array specifies tile widths (and may be empty)
+   * The number of widths specified is given by tile_width_count
+   */
+  int tile_widths[MAX_TILE_WIDTHS];
+
+/*!\brief Maximum number of tile heights in tile heights array.
+ *
+ * This define gives the maximum number of elements in the tile_heights array.
+ */
+#define MAX_TILE_HEIGHTS 64  // maximum tile height array length
+
+  /*!\brief Array of specified tile heights
+   *
+   * This array specifies tile heights (and may be empty)
+   * The number of heights specified is given by tile_height_count
+   */
+  int tile_heights[MAX_TILE_HEIGHTS];
 } aom_codec_enc_cfg_t; /**< alias for struct aom_codec_enc_cfg */
 
 /*!\brief Initialize an encoder instance
diff --git a/aom/aomcx.h b/aom/aomcx.h
index abe20cb..792802c 100644
--- a/aom/aomcx.h
+++ b/aom/aomcx.h
@@ -492,26 +492,6 @@
    */
   AV1E_SET_MTU,
 
-  /*!\brief Codec control function to set tile_width.
-   *
-   * In encoding this sets expilcit tiles of the given tile width
-   *
-   * By default, the value is 0 and explicit tiles are disabled.
-   *
-   * Experiment: MAX_TILE
-   */
-  AV1E_SET_TILE_WIDTH,
-
-  /*!\brief Codec control function to set tile_height.
-   *
-   * In encoding this sets expilcit tiles of the given tile height
-   *
-   * By default, the value is 0 and explicit tiles are disabled.
-   *
-   * Experiment: MAX_TILE
-   */
-  AV1E_SET_TILE_HEIGHT,
-
   /*!\brief Codec control function to set dependent_horz_tiles.
   *
   * In encoding and decoding, AV1 allows enabling dependent horizontal tile
diff --git a/aomenc.c b/aomenc.c
index b167eb3..99463a3 100644
--- a/aomenc.c
+++ b/aomenc.c
@@ -414,9 +414,9 @@
             "Number of tile rows to use, log2 (set to 0 while threads > 1)");
 #if CONFIG_MAX_TILE
 static const arg_def_t tile_width =
-    ARG_DEF(NULL, "tile-width", 1, "Width of each tile");
+    ARG_DEF(NULL, "tile-width", 1, "Tile widths (comma separated)");
 static const arg_def_t tile_height =
-    ARG_DEF(NULL, "tile-height", 1, "Height of each tile");
+    ARG_DEF(NULL, "tile-height", 1, "Tile heights (command separated)");
 #endif
 #if CONFIG_DEPENDENT_HORZTILES
 static const arg_def_t tile_dependent_rows =
@@ -553,10 +553,6 @@
 #endif  // CONFIG_EXT_TILE
                                        &tile_cols,
                                        &tile_rows,
-#if CONFIG_MAX_TILE
-                                       &tile_width,
-                                       &tile_height,
-#endif
 #if CONFIG_DEPENDENT_HORZTILES
                                        &tile_dependent_rows,
 #endif
@@ -614,10 +610,6 @@
 #endif  // CONFIG_EXT_TILE
                                         AV1E_SET_TILE_COLUMNS,
                                         AV1E_SET_TILE_ROWS,
-#if CONFIG_MAX_TILE
-                                        AV1E_SET_TILE_WIDTH,
-                                        AV1E_SET_TILE_HEIGHT,
-#endif
 #if CONFIG_DEPENDENT_HORZTILES
                                         AV1E_SET_TILE_DEPENDENT_ROWS,
 #endif
@@ -1119,6 +1111,14 @@
       config->cfg.kf_max_dist = arg_parse_uint(&arg);
     } else if (arg_match(&arg, &kf_disabled, argi)) {
       config->cfg.kf_mode = AOM_KF_DISABLED;
+#if CONFIG_MAX_TILE
+    } else if (arg_match(&arg, &tile_width, argi)) {
+      config->cfg.tile_width_count =
+          arg_parse_list(&arg, config->cfg.tile_widths, MAX_TILE_WIDTHS);
+    } else if (arg_match(&arg, &tile_height, argi)) {
+      config->cfg.tile_height_count =
+          arg_parse_list(&arg, config->cfg.tile_heights, MAX_TILE_HEIGHTS);
+#endif
     } else {
       int i, match = 0;
       for (i = 0; ctrl_args[i]; i++) {
diff --git a/args.c b/args.c
index 5711035..b9384de 100644
--- a/args.c
+++ b/args.c
@@ -210,3 +210,28 @@
   if (arg->def->enums) return arg_parse_enum(arg);
   return arg_parse_int(arg);
 }
+
+// parse a comma separated list of at most n integers
+// return the number of elements in the list
+int arg_parse_list(const struct arg *arg, int *list, int n) {
+  const char *ptr = arg->val;
+  char *endptr;
+  int i = 0;
+
+  while (ptr[0] != '\0') {
+    int32_t rawval = (int32_t)strtol(ptr, &endptr, 10);
+    if (rawval < INT_MIN || rawval > INT_MAX) {
+      die("Option %s: Value %ld out of range for signed int\n", arg->name,
+          rawval);
+    } else if (i >= n) {
+      die("Option %s: List has more than %d entries\n", arg->name, n);
+    } else if (*endptr == ',') {
+      endptr++;
+    } else if (*endptr != '\0') {
+      die("Option %s: Bad list separator '%c'\n", arg->name, *endptr);
+    }
+    list[i++] = (int)rawval;
+    ptr = endptr;
+  }
+  return i;
+}
diff --git a/args.h b/args.h
index e7841fc..c3427bc 100644
--- a/args.h
+++ b/args.h
@@ -57,6 +57,7 @@
 struct aom_rational arg_parse_rational(const struct arg *arg);
 int arg_parse_enum(const struct arg *arg);
 int arg_parse_enum_or_int(const struct arg *arg);
+int arg_parse_list(const struct arg *arg, int *list, int n);
 #ifdef __cplusplus
 }  // extern "C"
 #endif
diff --git a/av1/av1_cx_iface.c b/av1/av1_cx_iface.c
index a02193b..457c32c 100644
--- a/av1/av1_cx_iface.c
+++ b/av1/av1_cx_iface.c
@@ -36,10 +36,6 @@
   unsigned int static_thresh;
   unsigned int tile_columns;  // log2 number of tile columns
   unsigned int tile_rows;     // log2 number of tile rows
-#if CONFIG_MAX_TILE
-  unsigned int tile_width;   // tile width in superblocks  (if non zero)
-  unsigned int tile_height;  // tile height in superblocks (if non zero)
-#endif
 #if CONFIG_DEPENDENT_HORZTILES
   unsigned int dependent_horz_tiles;
 #endif
@@ -105,10 +101,6 @@
   0,    // static_thresh
   0,    // tile_columns
   0,    // tile_rows
-#if CONFIG_MAX_TILE
-  0,  // tile_width
-  0,  // tile_height
-#endif
 #if CONFIG_DEPENDENT_HORZTILES
   0,  // Dependent Horizontal tiles
 #endif
@@ -316,7 +308,6 @@
 #if CONFIG_MAX_TILE
     RANGE_CHECK_HI(extra_cfg, tile_columns, 6);
     RANGE_CHECK_HI(extra_cfg, tile_rows, 6);
-    RANGE_CHECK_HI(extra_cfg, tile_width, MAX_TILE_WIDTH_SB);
 #else   // CONFIG_MAX_TILE
   RANGE_CHECK_HI(extra_cfg, tile_columns, 6);
   RANGE_CHECK_HI(extra_cfg, tile_rows, 2);
@@ -639,8 +630,14 @@
 #endif  // CONFIG_EXT_TILE
 
 #if CONFIG_MAX_TILE
-  oxcf->tile_width = extra_cfg->tile_width;
-  oxcf->tile_height = extra_cfg->tile_height;
+  oxcf->tile_width_count = AOMMIN(cfg->tile_width_count, MAX_TILE_COLS);
+  oxcf->tile_height_count = AOMMIN(cfg->tile_height_count, MAX_TILE_ROWS);
+  for (int i = 0; i < oxcf->tile_width_count; i++) {
+    oxcf->tile_widths[i] = AOMMAX(cfg->tile_widths[i], 1);
+  }
+  for (int i = 0; i < oxcf->tile_height_count; i++) {
+    oxcf->tile_heights[i] = AOMMAX(cfg->tile_heights[i], 1);
+  }
 #endif
 #if CONFIG_DEPENDENT_HORZTILES
   oxcf->dependent_horz_tiles =
@@ -787,21 +784,6 @@
   return update_extra_cfg(ctx, &extra_cfg);
 }
 
-#if CONFIG_MAX_TILE
-static aom_codec_err_t ctrl_set_tile_width(aom_codec_alg_priv_t *ctx,
-                                           va_list args) {
-  struct av1_extracfg extra_cfg = ctx->extra_cfg;
-  extra_cfg.tile_width = CAST(AV1E_SET_TILE_WIDTH, args);
-  return update_extra_cfg(ctx, &extra_cfg);
-}
-
-static aom_codec_err_t ctrl_set_tile_height(aom_codec_alg_priv_t *ctx,
-                                            va_list args) {
-  struct av1_extracfg extra_cfg = ctx->extra_cfg;
-  extra_cfg.tile_height = CAST(AV1E_SET_TILE_HEIGHT, args);
-  return update_extra_cfg(ctx, &extra_cfg);
-}
-#endif
 #if CONFIG_DEPENDENT_HORZTILES
 static aom_codec_err_t ctrl_set_tile_dependent_rows(aom_codec_alg_priv_t *ctx,
                                                     va_list args) {
@@ -1582,10 +1564,6 @@
   { AOME_SET_STATIC_THRESHOLD, ctrl_set_static_thresh },
   { AV1E_SET_TILE_COLUMNS, ctrl_set_tile_columns },
   { AV1E_SET_TILE_ROWS, ctrl_set_tile_rows },
-#if CONFIG_MAX_TILE
-  { AV1E_SET_TILE_WIDTH, ctrl_set_tile_width },
-  { AV1E_SET_TILE_HEIGHT, ctrl_set_tile_height },
-#endif
 #if CONFIG_DEPENDENT_HORZTILES
   { AV1E_SET_TILE_DEPENDENT_ROWS, ctrl_set_tile_dependent_rows },
 #endif
@@ -1701,6 +1679,10 @@
         0,            // kf_min_dist
         9999,         // kf_max_dist
         0,            // large_scale_tile
+        0,            // tile_width_count
+        0,            // tile_height_count
+        { 0 },        // tile_widths
+        { 0 },        // tile_heights
     } },
 };
 
diff --git a/av1/encoder/encoder.c b/av1/encoder/encoder.c
index cc2d580..2de36de 100644
--- a/av1/encoder/encoder.c
+++ b/av1/encoder/encoder.c
@@ -927,18 +927,20 @@
   av1_get_tile_limits(cm);
 
   // configure tile columns
-  if (cpi->oxcf.tile_width == 0 || cpi->oxcf.tile_height == 0) {
+  if (cpi->oxcf.tile_width_count == 0 || cpi->oxcf.tile_height_count == 0) {
     cm->uniform_tile_spacing_flag = 1;
     cm->log2_tile_cols = AOMMAX(cpi->oxcf.tile_columns, cm->min_log2_tile_cols);
     cm->log2_tile_cols = AOMMIN(cm->log2_tile_cols, cm->max_log2_tile_cols);
   } else {
     int mi_cols = ALIGN_POWER_OF_TWO(cm->mi_cols, MAX_MIB_SIZE_LOG2);
     int sb_cols = mi_cols >> MAX_MIB_SIZE_LOG2;
-    int size_sb = AOMMIN(cpi->oxcf.tile_width, MAX_TILE_WIDTH_SB);
+    int size_sb, j = 0;
     cm->uniform_tile_spacing_flag = 0;
     for (i = 0, start_sb = 0; start_sb < sb_cols && i < MAX_TILE_COLS; i++) {
       cm->tile_col_start_sb[i] = start_sb;
-      start_sb += size_sb;
+      size_sb = cpi->oxcf.tile_widths[j++];
+      if (j >= cpi->oxcf.tile_width_count) j = 0;
+      start_sb += AOMMIN(size_sb, MAX_TILE_WIDTH_SB);
     }
     cm->tile_cols = i;
     cm->tile_col_start_sb[i] = sb_cols;
@@ -952,10 +954,12 @@
   } else {
     int mi_rows = ALIGN_POWER_OF_TWO(cm->mi_rows, MAX_MIB_SIZE_LOG2);
     int sb_rows = mi_rows >> MAX_MIB_SIZE_LOG2;
-    int size_sb = AOMMIN(cpi->oxcf.tile_height, cm->max_tile_height_sb);
+    int size_sb, j = 0;
     for (i = 0, start_sb = 0; start_sb < sb_rows && i < MAX_TILE_ROWS; i++) {
       cm->tile_row_start_sb[i] = start_sb;
-      start_sb += size_sb;
+      size_sb = cpi->oxcf.tile_heights[j++];
+      if (j >= cpi->oxcf.tile_height_count) j = 0;
+      start_sb += AOMMIN(size_sb, cm->max_tile_height_sb);
     }
     cm->tile_rows = i;
     cm->tile_row_start_sb[i] = sb_rows;
diff --git a/av1/encoder/encoder.h b/av1/encoder/encoder.h
index 84a7c26..27f96b5 100644
--- a/av1/encoder/encoder.h
+++ b/av1/encoder/encoder.h
@@ -266,8 +266,10 @@
   int tile_columns;
   int tile_rows;
 #if CONFIG_MAX_TILE
-  int tile_width;
-  int tile_height;
+  int tile_width_count;
+  int tile_height_count;
+  int tile_widths[MAX_TILE_COLS];
+  int tile_heights[MAX_TILE_ROWS];
 #endif
 #if CONFIG_DEPENDENT_HORZTILES
   int dependent_horz_tiles;