Coverage Report

Created: 2024-07-05 06:13

/src/pdf_fuzzer.cc
Line
Count
Source (jump to first uncovered line)
1
/*
2
# Copyright 2018 Google Inc.
3
#
4
# Licensed under the Apache License, Version 2.0 (the "License");
5
# you may not use this file except in compliance with the License.
6
# You may obtain a copy of the License at
7
#
8
#      http://www.apache.org/licenses/LICENSE-2.0
9
#
10
# Unless required by applicable law or agreed to in writing, software
11
# distributed under the License is distributed on an "AS IS" BASIS,
12
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13
# See the License for the specific language governing permissions and
14
# limitations under the License.
15
#
16
################################################################################
17
*/
18
19
#include <cstdint>
20
#include <stdlib.h>
21
#include <string.h>
22
#include <inttypes.h>
23
24
#include <mupdf/fitz.h>
25
26
394M
#define ALIGNMENT ((size_t) 16)
27
49.0M
#define KBYTE ((size_t) 1024)
28
49.0M
#define MBYTE (1024 * KBYTE)
29
49.0M
#define GBYTE (1024 * MBYTE)
30
49.0M
#define MAX_ALLOCATION (1 * GBYTE)
31
32
static size_t used;
33
34
static void *fz_limit_reached_ossfuzz(size_t oldsize, size_t size)
35
76
{
36
76
  if (oldsize == 0)
37
76
    fprintf(stderr, "limit: %zu Mbyte used: %zu Mbyte allocation: %zu: limit reached\n", MAX_ALLOCATION / MBYTE, used / MBYTE, size);
38
0
  else
39
0
    fprintf(stderr, "limit: %zu Mbyte used: %zu Mbyte reallocation: %zu -> %zu: limit reached\n", MAX_ALLOCATION / MBYTE, used / MBYTE, oldsize, size);
40
76
  fflush(0);
41
76
  return NULL;
42
76
}
43
44
static void *fz_malloc_ossfuzz(void *opaque, size_t size)
45
47.2M
{
46
47.2M
  char *ptr = NULL;
47
48
47.2M
  if (size == 0)
49
0
    return NULL;
50
47.2M
  if (size > SIZE_MAX - ALIGNMENT)
51
0
    return NULL;
52
47.2M
  if (size + ALIGNMENT > MAX_ALLOCATION - used)
53
76
    return fz_limit_reached_ossfuzz(0, size + ALIGNMENT);
54
55
47.2M
  ptr = (char *) malloc(size + ALIGNMENT);
56
47.2M
  if (ptr == NULL)
57
0
    return NULL;
58
59
47.2M
  memcpy(ptr, &size, sizeof(size));
60
47.2M
  used += size + ALIGNMENT;
61
62
47.2M
  return ptr + ALIGNMENT;
63
47.2M
}
64
65
static void fz_free_ossfuzz(void *opaque, void *ptr)
66
47.2M
{
67
47.2M
  size_t size;
68
69
47.2M
  if (ptr == NULL)
70
0
    return;
71
47.2M
  if (ptr < (void *) ALIGNMENT)
72
0
    return;
73
74
47.2M
  ptr = (char *) ptr - ALIGNMENT;
75
47.2M
  memcpy(&size, ptr, sizeof(size));
76
77
47.2M
  used -= size + ALIGNMENT;
78
47.2M
  free(ptr);
79
47.2M
}
80
81
static void *fz_realloc_ossfuzz(void *opaque, void *old, size_t size)
82
3.51M
{
83
3.51M
  size_t oldsize;
84
3.51M
  char *ptr;
85
86
3.51M
  if (old == NULL)
87
1.68M
    return fz_malloc_ossfuzz(opaque, size);
88
1.83M
  if (old < (void *) ALIGNMENT)
89
0
    return NULL;
90
91
1.83M
  if (size == 0) {
92
0
    fz_free_ossfuzz(opaque, old);
93
0
    return NULL;
94
0
  }
95
1.83M
  if (size > SIZE_MAX - ALIGNMENT)
96
0
    return NULL;
97
98
1.83M
  old = (char *) old - ALIGNMENT;
99
1.83M
  memcpy(&oldsize, old, sizeof(oldsize));
100
101
1.83M
  if (size + ALIGNMENT > MAX_ALLOCATION - used + oldsize + ALIGNMENT)
102
0
    return fz_limit_reached_ossfuzz(oldsize + ALIGNMENT, size + ALIGNMENT);
103
104
1.83M
  ptr = (char *) realloc(old, size + ALIGNMENT);
105
1.83M
  if (ptr == NULL)
106
0
    return NULL;
107
108
1.83M
  used -= oldsize + ALIGNMENT;
109
1.83M
  memcpy(ptr, &size, sizeof(size));
110
1.83M
  used += size + ALIGNMENT;
111
112
1.83M
  return ptr + ALIGNMENT;
113
1.83M
}
114
115
static fz_alloc_context fz_alloc_ossfuzz =
116
{
117
  NULL,
118
  fz_malloc_ossfuzz,
119
  fz_realloc_ossfuzz,
120
  fz_free_ossfuzz
121
};
122
123
8.79k
extern "C" int LLVMFuzzerTestOneInput(const uint8_t *data, size_t size) {
124
8.79k
  fz_context *ctx;
125
8.79k
  fz_stream *stream;
126
8.79k
  fz_document *doc;
127
8.79k
  fz_pixmap *pix;
128
129
8.79k
  used = 0;
130
131
8.79k
  ctx = fz_new_context(&fz_alloc_ossfuzz, nullptr, FZ_STORE_DEFAULT);
132
8.79k
  stream = NULL;
133
8.79k
  doc = NULL;
134
8.79k
  pix = NULL;
135
136
8.79k
  fz_var(stream);
137
8.79k
  fz_var(doc);
138
8.79k
  fz_var(pix);
139
140
17.5k
  fz_try(ctx) {
141
8.79k
    fz_register_document_handlers(ctx);
142
8.79k
    stream = fz_open_memory(ctx, data, size);
143
8.79k
    doc = fz_open_document_with_stream(ctx, "pdf", stream);
144
145
18.2k
    for (int i = 0; i < fz_count_pages(ctx, doc); i++) {
146
9.44k
      pix = fz_new_pixmap_from_page_number(ctx, doc, i, fz_identity, fz_device_rgb(ctx), 0);
147
9.44k
      fz_drop_pixmap(ctx, pix);
148
9.44k
      pix = NULL;
149
9.44k
    }
150
8.79k
  }
151
17.5k
  fz_always(ctx) {
152
8.79k
    fz_drop_pixmap(ctx, pix);
153
8.79k
    fz_drop_document(ctx, doc);
154
8.79k
    fz_drop_stream(ctx, stream);
155
8.79k
  }
156
8.79k
  fz_catch(ctx) {
157
1.16k
    fz_report_error(ctx);
158
1.16k
    fz_log_error(ctx, "error rendering pages");
159
1.16k
  }
160
161
8.79k
  fz_flush_warnings(ctx);
162
8.79k
  fz_drop_context(ctx);
163
164
8.79k
  return 0;
165
8.79k
}