Coverage Report

Created: 2025-04-22 06:20

/src/libspectre/ghostscript/base/gsnotify.c
Line
Count
Source (jump to first uncovered line)
1
/* Copyright (C) 2001-2020 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.,  1305 Grant Avenue - Suite 200, Novato,
13
   CA 94945, U.S.A., +1(415)492-9861, for further information.
14
*/
15
16
17
/* Notification machinery implementation */
18
#include "gx.h"
19
#include "gserrors.h"
20
#include "gsstruct.h"
21
#include "gsnotify.h"
22
23
/* GC descriptors */
24
private_st_gs_notify_registration();
25
public_st_gs_notify_list();
26
27
/* Initialize a notification list. */
28
void
29
gs_notify_init(gs_notify_list_t *nlist, gs_memory_t *mem)
30
218
{
31
218
    nlist->first = 0;
32
218
    nlist->memory = mem;
33
218
}
34
35
/* Register a client. */
36
int
37
gs_notify_register(gs_notify_list_t *nlist, gs_notify_proc_t proc,
38
                   void *proc_data)
39
0
{
40
0
    gs_notify_registration_t *nreg =
41
0
        gs_alloc_struct(nlist->memory, gs_notify_registration_t,
42
0
                        &st_gs_notify_registration, "gs_notify_register");
43
44
0
    if (nreg == 0)
45
0
        return_error(gs_error_VMerror);
46
0
    nreg->proc = proc;
47
0
    nreg->proc_data = proc_data;
48
0
    nreg->next = nlist->first;
49
0
    nlist->first = nreg;
50
0
    return 0;
51
0
}
52
53
/*
54
 * Unregister a client.  Return 1 if the client was registered, 0 if not.
55
 * If proc_data is 0, unregister all registrations of that proc; otherwise,
56
 * unregister only the registration of that procedure with that proc_data.
57
 */
58
static void
59
no_unreg_proc(void *pdata)
60
0
{
61
0
}
62
int
63
gs_notify_unregister_calling(gs_notify_list_t *nlist, gs_notify_proc_t proc,
64
                             void *proc_data,
65
                             void (*unreg_proc)(void *pdata))
66
0
{
67
0
    gs_notify_registration_t **prev = &nlist->first;
68
0
    gs_notify_registration_t *cur;
69
0
    bool found = 0;
70
71
0
    while ((cur = *prev) != 0)
72
0
        if (cur->proc == proc &&
73
0
            (proc_data == 0 || cur->proc_data == proc_data)
74
0
            ) {
75
0
            *prev = cur->next;
76
0
            unreg_proc(cur->proc_data);
77
0
            gs_free_object(nlist->memory, cur, "gs_notify_unregister");
78
0
            found = 1;
79
0
        } else
80
0
            prev = &cur->next;
81
0
    return found;
82
0
}
83
int
84
gs_notify_unregister(gs_notify_list_t *nlist, gs_notify_proc_t proc,
85
                     void *proc_data)
86
0
{
87
0
    return gs_notify_unregister_calling(nlist, proc, proc_data, no_unreg_proc);
88
0
}
89
90
/*
91
 * Notify the clients on a list.  If an error occurs, return the first
92
 * error code, but notify all clients regardless.
93
 */
94
int
95
gs_notify_all(gs_notify_list_t *nlist, void *event_data)
96
218
{
97
218
    gs_notify_registration_t *cur;
98
218
    gs_notify_registration_t *next;
99
218
    int ecode = 0;
100
101
218
    for (next = nlist->first; (cur = next) != 0;) {
102
0
        int code;
103
104
0
        next = cur->next;
105
0
        code = cur->proc(cur->proc_data, event_data);
106
0
        if (code < 0 && ecode == 0)
107
0
            ecode = code;
108
0
    }
109
218
    return ecode;
110
218
}
111
112
/* Release a notification list. */
113
void
114
gs_notify_release(gs_notify_list_t *nlist)
115
218
{
116
218
    gs_memory_t *mem = nlist->memory;
117
118
218
    while (nlist->first) {
119
0
        gs_notify_registration_t *next = nlist->first->next;
120
121
0
        gs_free_object(mem, nlist->first, "gs_notify_release");
122
0
        nlist->first = next;
123
0
    }
124
218
}