Blame imageio/imageio_util.c

Packit 9c6abc
// Copyright 2016 Google Inc. All Rights Reserved.
Packit 9c6abc
//
Packit 9c6abc
// Use of this source code is governed by a BSD-style license
Packit 9c6abc
// that can be found in the COPYING file in the root of the source
Packit 9c6abc
// tree. An additional intellectual property rights grant can be found
Packit 9c6abc
// in the file PATENTS. All contributing project authors may
Packit 9c6abc
// be found in the AUTHORS file in the root of the source tree.
Packit 9c6abc
// -----------------------------------------------------------------------------
Packit 9c6abc
//
Packit 9c6abc
//  Utility functions used by the image decoders.
Packit 9c6abc
//
Packit 9c6abc
Packit 9c6abc
#include "./imageio_util.h"
Packit 9c6abc
Packit 9c6abc
#if defined(_WIN32)
Packit 9c6abc
#include <fcntl.h>   // for _O_BINARY
Packit 9c6abc
#include <io.h>      // for _setmode()
Packit 9c6abc
#endif
Packit 9c6abc
#include <stdlib.h>
Packit 9c6abc
#include <string.h>
Packit 9c6abc
Packit 9c6abc
// -----------------------------------------------------------------------------
Packit 9c6abc
// File I/O
Packit 9c6abc
Packit 9c6abc
FILE* ImgIoUtilSetBinaryMode(FILE* file) {
Packit 9c6abc
#if defined(_WIN32)
Packit 9c6abc
  if (_setmode(_fileno(file), _O_BINARY) == -1) {
Packit 9c6abc
    fprintf(stderr, "Failed to reopen file in O_BINARY mode.\n");
Packit 9c6abc
    return NULL;
Packit 9c6abc
  }
Packit 9c6abc
#endif
Packit 9c6abc
  return file;
Packit 9c6abc
}
Packit 9c6abc
Packit 9c6abc
int ImgIoUtilReadFromStdin(const uint8_t** data, size_t* data_size) {
Packit 9c6abc
  static const size_t kBlockSize = 16384;  // default initial size
Packit 9c6abc
  size_t max_size = 0;
Packit 9c6abc
  size_t size = 0;
Packit 9c6abc
  uint8_t* input = NULL;
Packit 9c6abc
Packit 9c6abc
  if (data == NULL || data_size == NULL) return 0;
Packit 9c6abc
  *data = NULL;
Packit 9c6abc
  *data_size = 0;
Packit 9c6abc
Packit 9c6abc
  if (!ImgIoUtilSetBinaryMode(stdin)) return 0;
Packit 9c6abc
Packit 9c6abc
  while (!feof(stdin)) {
Packit 9c6abc
    // We double the buffer size each time and read as much as possible.
Packit 9c6abc
    const size_t extra_size = (max_size == 0) ? kBlockSize : max_size;
Packit 9c6abc
    // we allocate one extra byte for the \0 terminator
Packit 9c6abc
    void* const new_data = realloc(input, max_size + extra_size + 1);
Packit 9c6abc
    if (new_data == NULL) goto Error;
Packit 9c6abc
    input = (uint8_t*)new_data;
Packit 9c6abc
    max_size += extra_size;
Packit 9c6abc
    size += fread(input + size, 1, extra_size, stdin);
Packit 9c6abc
    if (size < max_size) break;
Packit 9c6abc
  }
Packit 9c6abc
  if (ferror(stdin)) goto Error;
Packit 9c6abc
  if (input != NULL) input[size] = '\0';  // convenient 0-terminator
Packit 9c6abc
  *data = input;
Packit 9c6abc
  *data_size = size;
Packit 9c6abc
  return 1;
Packit 9c6abc
Packit 9c6abc
 Error:
Packit 9c6abc
  free(input);
Packit 9c6abc
  fprintf(stderr, "Could not read from stdin\n");
Packit 9c6abc
  return 0;
Packit 9c6abc
}
Packit 9c6abc
Packit 9c6abc
int ImgIoUtilReadFile(const char* const file_name,
Packit 9c6abc
                      const uint8_t** data, size_t* data_size) {
Packit 9c6abc
  int ok;
Packit 9c6abc
  uint8_t* file_data;
Packit 9c6abc
  size_t file_size;
Packit 9c6abc
  FILE* in;
Packit 9c6abc
  const int from_stdin = (file_name == NULL) || !strcmp(file_name, "-");
Packit 9c6abc
Packit 9c6abc
  if (from_stdin) return ImgIoUtilReadFromStdin(data, data_size);
Packit 9c6abc
Packit 9c6abc
  if (data == NULL || data_size == NULL) return 0;
Packit 9c6abc
  *data = NULL;
Packit 9c6abc
  *data_size = 0;
Packit 9c6abc
Packit 9c6abc
  in = fopen(file_name, "rb");
Packit 9c6abc
  if (in == NULL) {
Packit 9c6abc
    fprintf(stderr, "cannot open input file '%s'\n", file_name);
Packit 9c6abc
    return 0;
Packit 9c6abc
  }
Packit 9c6abc
  fseek(in, 0, SEEK_END);
Packit 9c6abc
  file_size = ftell(in);
Packit 9c6abc
  fseek(in, 0, SEEK_SET);
Packit 9c6abc
  // we allocate one extra byte for the \0 terminator
Packit 9c6abc
  file_data = (uint8_t*)malloc(file_size + 1);
Packit 9c6abc
  if (file_data == NULL) {
Packit 9c6abc
    fclose(in);
Packit 9c6abc
    fprintf(stderr, "memory allocation failure when reading file %s\n",
Packit 9c6abc
            file_name);
Packit 9c6abc
    return 0;
Packit 9c6abc
  }
Packit 9c6abc
  ok = (fread(file_data, file_size, 1, in) == 1);
Packit 9c6abc
  fclose(in);
Packit 9c6abc
Packit 9c6abc
  if (!ok) {
Packit 9c6abc
    fprintf(stderr, "Could not read %d bytes of data from file %s\n",
Packit 9c6abc
            (int)file_size, file_name);
Packit 9c6abc
    free(file_data);
Packit 9c6abc
    return 0;
Packit 9c6abc
  }
Packit 9c6abc
  file_data[file_size] = '\0';  // convenient 0-terminator
Packit 9c6abc
  *data = file_data;
Packit 9c6abc
  *data_size = file_size;
Packit 9c6abc
  return 1;
Packit 9c6abc
}
Packit 9c6abc
Packit 9c6abc
// -----------------------------------------------------------------------------
Packit 9c6abc
Packit 9c6abc
int ImgIoUtilWriteFile(const char* const file_name,
Packit 9c6abc
                       const uint8_t* data, size_t data_size) {
Packit 9c6abc
  int ok;
Packit 9c6abc
  FILE* out;
Packit 9c6abc
  const int to_stdout = (file_name == NULL) || !strcmp(file_name, "-");
Packit 9c6abc
Packit 9c6abc
  if (data == NULL) {
Packit 9c6abc
    return 0;
Packit 9c6abc
  }
Packit 9c6abc
  out = to_stdout ? ImgIoUtilSetBinaryMode(stdout) : fopen(file_name, "wb");
Packit 9c6abc
  if (out == NULL) {
Packit 9c6abc
    fprintf(stderr, "Error! Cannot open output file '%s'\n", file_name);
Packit 9c6abc
    return 0;
Packit 9c6abc
  }
Packit 9c6abc
  ok = (fwrite(data, data_size, 1, out) == 1);
Packit 9c6abc
  if (out != stdout) fclose(out);
Packit 9c6abc
  return ok;
Packit 9c6abc
}
Packit 9c6abc
Packit 9c6abc
// -----------------------------------------------------------------------------
Packit 9c6abc
Packit 9c6abc
void ImgIoUtilCopyPlane(const uint8_t* src, int src_stride,
Packit 9c6abc
                        uint8_t* dst, int dst_stride, int width, int height) {
Packit 9c6abc
  while (height-- > 0) {
Packit 9c6abc
    memcpy(dst, src, width * sizeof(*dst));
Packit 9c6abc
    src += src_stride;
Packit 9c6abc
    dst += dst_stride;
Packit 9c6abc
  }
Packit 9c6abc
}
Packit 9c6abc
Packit 9c6abc
// -----------------------------------------------------------------------------
Packit 9c6abc
Packit 9c6abc
int ImgIoUtilCheckSizeArgumentsOverflow(uint64_t nmemb, size_t size) {
Packit 9c6abc
  const uint64_t total_size = nmemb * size;
Packit 9c6abc
  int ok = (total_size == (size_t)total_size);
Packit 9c6abc
#if defined(WEBP_MAX_IMAGE_SIZE)
Packit 9c6abc
  ok = ok && (total_size <= (uint64_t)WEBP_MAX_IMAGE_SIZE);
Packit 9c6abc
#endif
Packit 9c6abc
  return ok;
Packit 9c6abc
}
Packit 9c6abc
Packit 9c6abc
// -----------------------------------------------------------------------------