| #include <libexif/exif-data.h> |
| #include <libexif/exif-loader.h> |
| #include <stddef.h> |
| #include <stdlib.h> |
| #include <string.h> |
| |
| constexpr size_t kMaxDataSize = 32; |
| |
| /* Extract all MakerNote tags */ |
| static void mnote_dump(ExifData *data) { |
| ExifMnoteData *mn = exif_data_get_mnote_data(data); |
| if (mn) { |
| int num = exif_mnote_data_count(mn); |
| |
| /* Loop through all MakerNote tags */ |
| for (int i = 0; i < num; ++i) { |
| char buf[1024]; |
| exif_mnote_data_get_value(mn, i, buf, sizeof(buf)); |
| } |
| } |
| } |
| |
| static void dump_value(ExifEntry *entry, void *) { |
| char buf[1024]; |
| exif_entry_get_value(entry, buf, sizeof(buf)); |
| } |
| |
| static void data_func(ExifContent *content, void *) { |
| exif_content_foreach_entry(content, dump_value, NULL); |
| } |
| |
| /* This is like exif_data_dump but without writing to stdout */ |
| static void data_dump(ExifData *data) { |
| exif_data_foreach_content(data, data_func, NULL); |
| } |
| |
| extern "C" int LLVMFuzzerTestOneInput(const uint8_t *data, size_t size) { |
| if (!data) { |
| return 0; |
| } |
| |
| size = (size % kMaxDataSize); |
| uint8_t *buffer = (uint8_t *)malloc(size * sizeof(uint8_t)); |
| memcpy(buffer, data, size); |
| // Parse tags using (ultimately) exif_data_load_data() |
| auto image = exif_data_new_from_data(buffer, size); |
| if (image) { |
| // Exercise the EXIF tag manipulation code |
| exif_data_get_mnote_data(image); |
| data_dump(image); |
| mnote_dump(image); |
| unsigned char *buf; |
| unsigned int sz; |
| exif_data_save_data(image, &buf, &sz); |
| free(buf); |
| exif_data_fix(image); |
| exif_data_unref(image); |
| } |
| |
| // Parse tags again, but using exif_loader_write(), which is a separate |
| // parser. There is no need to fuzz the parsed ExifData again, since it will |
| // be identical to what has been loaded (and fuzzed) above. |
| ExifLoader *loader = exif_loader_new(); |
| if (!loader) { |
| free(buffer); |
| return 0; |
| } |
| exif_loader_write(loader, const_cast<unsigned char *>(buffer), size); |
| image = exif_loader_get_data(loader); |
| if (image) { |
| exif_data_unref(image); |
| } |
| exif_loader_unref(loader); |
| free(buffer); |
| return 0; |
| } |