Coverage Report

Created: 2025-11-16 07:40

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/ghostpdl/pdf/pdf_loop_detect.c
Line
Count
Source
1
/* Copyright (C) 2018-2025 Artifex Software, Inc.
2
   All Rights Reserved.
3
4
   This software is provided AS-IS with no warranty, either express or
5
   implied.
6
7
   This software is distributed under license and may not be copied,
8
   modified or distributed except as expressly authorized under the terms
9
   of the license contained in the file LICENSE in this distribution.
10
11
   Refer to licensing information at http://www.artifex.com or contact
12
   Artifex Software, Inc.,  39 Mesa Street, Suite 108A, San Francisco,
13
   CA 94129, USA, for further information.
14
*/
15
16
/* code for handling circular references */
17
18
#include "pdf_int.h"
19
#include "pdf_loop_detect.h"
20
21
static int pdfi_init_loop_detector(pdf_context *ctx)
22
12.0M
{
23
12.0M
    if (ctx->loop_detection) {
24
0
        dbgmprintf(ctx->memory, "Attempt to initialise loop detector while one is in operation\n");
25
0
        return_error(gs_error_unknownerror);
26
0
    }
27
28
12.0M
    ctx->loop_detection = (uint64_t *)gs_alloc_bytes(ctx->memory, INITIAL_LOOP_TRACKER_SIZE * sizeof (uint64_t), "allocate loop tracking array");
29
12.0M
    if (ctx->loop_detection == NULL)
30
0
        return_error(gs_error_VMerror);
31
32
12.0M
    ctx->loop_detection_entries = 0;
33
12.0M
    ctx->loop_detection_size = INITIAL_LOOP_TRACKER_SIZE;
34
12.0M
    return 0;
35
12.0M
}
36
37
static int pdfi_free_loop_detector(pdf_context *ctx)
38
12.0M
{
39
12.0M
    if (ctx->loop_detection == NULL) {
40
0
        dbgmprintf(ctx->memory, "Attempt to free loop detector without initialising it\n");
41
0
        return 0;
42
0
    }
43
12.0M
    if (ctx->loop_detection != NULL)
44
12.0M
        gs_free_object(ctx->memory, ctx->loop_detection, "Free array for loop tracking");
45
12.0M
    ctx->loop_detection_entries = 0;
46
12.0M
    ctx->loop_detection_size = 0;
47
12.0M
    ctx->loop_detection = NULL;
48
49
12.0M
    return 0;
50
12.0M
}
51
52
static int pdfi_loop_detector_add_object_unchecked(pdf_context *ctx, uint64_t object)
53
37.5M
{
54
37.5M
    if (ctx->loop_detection == NULL) {
55
0
        dbgmprintf(ctx->memory, "Attempt to use loop detector without initialising it\n");
56
0
        return 0;
57
0
    }
58
59
37.5M
    if (ctx->loop_detection_entries == ctx->loop_detection_size) {
60
28
        uint64_t *New;
61
62
28
        New = (uint64_t *)gs_alloc_bytes(ctx->memory, (size_t)(ctx->loop_detection_size + INITIAL_LOOP_TRACKER_SIZE) * (size_t)sizeof (uint64_t), "re-allocate loop tracking array");
63
28
        if (New == NULL) {
64
0
            return_error(gs_error_VMerror);
65
0
        }
66
28
        memcpy(New, ctx->loop_detection, ctx->loop_detection_entries * sizeof(uint64_t));
67
28
        gs_free_object(ctx->memory, ctx->loop_detection, "Free array for loop tracking");
68
28
        ctx->loop_detection_size += INITIAL_LOOP_TRACKER_SIZE;
69
28
        ctx->loop_detection = New;
70
28
    }
71
37.5M
    ctx->loop_detection[ctx->loop_detection_entries++] = object;
72
37.5M
    return 0;
73
37.5M
}
74
75
int pdfi_loop_detector_add_object(pdf_context *ctx, uint64_t object)
76
13.8M
{
77
13.8M
    if (object == 0) {
78
0
        dbgmprintf(ctx->memory, "Attempt to add an object number of 0 to the loop detection\n");
79
0
        return 0;
80
0
    }
81
13.8M
    return pdfi_loop_detector_add_object_unchecked(ctx, object);
82
13.8M
}
83
84
bool pdfi_loop_detector_check_object(pdf_context *ctx, uint64_t object)
85
10.4M
{
86
10.4M
    int i = 0;
87
88
10.4M
    if (ctx->loop_detection == NULL) {
89
16
        dbgmprintf(ctx->memory, "Attempt to use loop detector without initialising it\n");
90
16
        return 0;
91
16
    }
92
93
29.0M
    for (i=0;i < ctx->loop_detection_entries;i++) {
94
18.7M
        if (ctx->loop_detection[i] == object) {
95
44.1k
            char info_string[256];
96
44.1k
            gs_snprintf(info_string, sizeof(info_string), "Error! circular reference to object %"PRIu64" detected.\n", object);
97
44.1k
            pdfi_set_error(ctx, 0, NULL, E_PDF_CIRCULARREF, "pdfi_loop_detector_check_object", info_string);
98
44.1k
            return true;
99
44.1k
        }
100
18.7M
    }
101
10.3M
    return false;
102
10.4M
}
103
104
int pdfi_loop_detector_mark(pdf_context *ctx)
105
23.7M
{
106
23.7M
    int code = 0;
107
108
23.7M
    if (ctx->loop_detection == NULL) {
109
12.0M
        code = pdfi_init_loop_detector(ctx);
110
12.0M
        if (code < 0)
111
0
            return code;
112
12.0M
    }
113
114
23.7M
    return pdfi_loop_detector_add_object_unchecked(ctx, 0);
115
23.7M
}
116
117
int pdfi_loop_detector_cleartomark(pdf_context *ctx)
118
23.7M
{
119
23.7M
    if (ctx->loop_detection == NULL) {
120
0
        dbgmprintf(ctx->memory, "Attempt to use loop detector without initialising it\n");
121
0
        return 0;
122
0
    }
123
124
37.5M
    while (ctx->loop_detection[--ctx->loop_detection_entries] != 0) {
125
13.8M
        ctx->loop_detection[ctx->loop_detection_entries] = 0;
126
13.8M
    }
127
    /* FIXME - potential optimisation
128
     * Instead of freeing the loop detection array every tiome we are done with it
129
     * and then reallocating a new one next time we need one, we could just keep
130
     * the existing (empty) array. I suspect this would provide a small performance
131
     * improvement.
132
     */
133
23.7M
    if (ctx->loop_detection_entries == 0)
134
12.0M
        pdfi_free_loop_detector(ctx);
135
23.7M
    return 0;
136
23.7M
}