Add oss-fuzz implementation to tests/
diff --git a/tests/oss-fuzz/avif_decode_fuzzer.cc b/tests/oss-fuzz/avif_decode_fuzzer.cc
new file mode 100644
index 0000000..8e7edfc
--- /dev/null
+++ b/tests/oss-fuzz/avif_decode_fuzzer.cc
@@ -0,0 +1,89 @@
+// Copyright 2020 Google Inc.
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+//
+//      http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+//
+//###############################################################################
+
+#include "avif/avif.h"
+
+extern "C" int LLVMFuzzerTestOneInput(const uint8_t *Data, size_t Size) {
+  static avifRGBFormat rgbFormats[] = {AVIF_RGB_FORMAT_RGB,
+                                       AVIF_RGB_FORMAT_RGBA};
+  static size_t rgbFormatsCount = sizeof(rgbFormats) / sizeof(rgbFormats[0]);
+
+  static avifChromaUpsampling upsamplings[] = {AVIF_CHROMA_UPSAMPLING_BILINEAR,
+                                               AVIF_CHROMA_UPSAMPLING_NEAREST};
+  static size_t upsamplingsCount = sizeof(upsamplings) / sizeof(upsamplings[0]);
+
+  static uint32_t rgbDepths[] = {8, 10};
+  static size_t rgbDepthsCount = sizeof(rgbDepths) / sizeof(rgbDepths[0]);
+
+  static uint32_t yuvDepths[] = {8, 10};
+  static size_t yuvDepthsCount = sizeof(yuvDepths) / sizeof(yuvDepths[0]);
+
+  avifDecoder *decoder = avifDecoderCreate();
+  avifResult result = avifDecoderSetIOMemory(decoder, Data, Size);
+  if (result == AVIF_RESULT_OK) {
+    result = avifDecoderParse(decoder);
+  }
+  if (result == AVIF_RESULT_OK) {
+    for (int loop = 0; loop < 2; ++loop) {
+      while (avifDecoderNextImage(decoder) == AVIF_RESULT_OK) {
+        avifRGBImage rgb;
+        avifRGBImageSetDefaults(&rgb, decoder->image);
+
+        for (size_t rgbFormatsIndex = 0; rgbFormatsIndex < rgbFormatsCount;
+             ++rgbFormatsIndex) {
+          for (size_t upsamplingsIndex = 0; upsamplingsIndex < upsamplingsCount;
+               ++upsamplingsIndex) {
+            for (size_t rgbDepthsIndex = 0; rgbDepthsIndex < rgbDepthsCount;
+                 ++rgbDepthsIndex) {
+              // Convert to RGB
+              rgb.format = rgbFormats[rgbFormatsIndex];
+              rgb.depth = rgbDepths[rgbDepthsIndex];
+              rgb.chromaUpsampling = upsamplings[upsamplingsIndex];
+              avifRGBImageAllocatePixels(&rgb);
+              avifResult rgbResult = avifImageYUVToRGB(decoder->image, &rgb);
+              if (rgbResult == AVIF_RESULT_OK) {
+                for (size_t yuvDepthsIndex = 0; yuvDepthsIndex < yuvDepthsCount;
+                     ++yuvDepthsIndex) {
+                  // ... and back to YUV
+                  avifImage *tempImage = avifImageCreate(
+                      decoder->image->width, decoder->image->height,
+                      yuvDepths[yuvDepthsIndex], decoder->image->yuvFormat);
+                  avifResult yuvResult = avifImageRGBToYUV(tempImage, &rgb);
+                  if (yuvResult != AVIF_RESULT_OK) {
+                  }
+                  avifImageDestroy(tempImage);
+                }
+              }
+
+              avifRGBImageFreePixels(&rgb);
+            }
+          }
+        }
+      }
+
+      if (loop != 1) {
+        result = avifDecoderReset(decoder);
+        if (result == AVIF_RESULT_OK) {
+        } else {
+          break;
+        }
+      }
+    }
+  }
+
+  avifDecoderDestroy(decoder);
+  return 0; // Non-zero return values are reserved for future use.
+}