Cthulhu  0.2.10
Cthulhu compiler collection
notify.c
Go to the documentation of this file.
1 // SPDX-License-Identifier: LGPL-3.0-only
2 
3 #include "notify/notify.h"
4 
5 #include "base/panic.h"
6 #include "arena/arena.h"
7 
8 #include "std/set.h"
9 #include "std/str.h"
10 #include "std/typed/vector.h"
11 #include "std/vector.h"
12 
13 typedef struct logger_t
14 {
16 
18 } logger_t;
19 
22 {
23  CTASSERT(arena != NULL);
24 
25  logger_t *logs = ARENA_MALLOC(sizeof(logger_t), "logger", NULL, arena);
26 
27  logs->arena = arena;
28  logs->messages = typevec_new(sizeof(event_t), 8, arena);
29 
30  ARENA_IDENTIFY(logs->messages, "messages", logs, arena);
31 
32  return logs;
33 }
34 
37 {
38  CTASSERT(logs != NULL);
39 
40  return logs->messages;
41 }
42 
44 bool logger_has_errors(const logger_t *logs, notify_rules_t rules)
45 {
46  CTASSERT(logs != NULL);
47  CTASSERT(rules.ignored_warnings != NULL);
48  CTASSERT(rules.warnings_as_errors != NULL);
49 
50  const typevec_t *events = logger_get_events(logs);
51  size_t len = typevec_len(events);
52 
53  for (size_t i = 0; i < len; i++)
54  {
55  event_t event;
56  typevec_get(events, i, &event);
57 
58  const diagnostic_t *diagnostic = event.diagnostic;
59  CTASSERTF(diagnostic != NULL, "event %zu has no diagnostic", i);
60 
61  switch (diagnostic->severity)
62  {
63  case eSeverityFatal:
64  case eSeverityInternal:
65  case eSeveritySorry:
66  return true;
67 
68  case eSeverityWarn:
69  if (set_contains(rules.ignored_warnings, diagnostic))
70  {
71  continue;
72  }
73 
74  if (set_contains(rules.warnings_as_errors, diagnostic))
75  {
76  return true;
77  }
78  break;
79 
80  default:
81  break;
82  }
83  }
84 
85  return false;
86 }
87 
90 {
91  CTASSERT(logs != NULL);
92 
93  typevec_reset(logs->messages);
94 }
95 
98 {
99  CTASSERT(logs != NULL);
100 
101  return logs->arena;
102 }
103 
104 STA_DECL
105 event_builder_t msg_notify(logger_t *logs, const diagnostic_t *diagnostic, const node_t *node, const char *fmt, ...)
106 {
107  va_list args;
108  va_start(args, fmt);
109 
110  event_builder_t event = msg_vnotify(logs, diagnostic, node, fmt, args);
111 
112  va_end(args);
113 
114  return event;
115 }
116 
117 STA_DECL
118 event_builder_t msg_vnotify(logger_t *logs, const diagnostic_t *diagnostic, const node_t *node, const char *fmt, va_list args)
119 {
120  CTASSERT(logs != NULL);
121  CTASSERT(diagnostic != NULL);
122  CTASSERT(node != NULL);
123 
124  char *msg = str_vformat(logs->arena, fmt, args);
125 
126  event_t event = {
127  .diagnostic = diagnostic,
128  .node = *node,
129  .message = msg,
130  .segments = NULL,
131  .notes = NULL,
132  };
133 
134  event_t *result = typevec_push(logs->messages, &event);
135  event_builder_t builder = {
136  .event = result,
137  .arena = logs->arena
138  };
139  return builder;
140 }
141 
142 STA_DECL
143 void msg_append(event_builder_t builder, const node_t *node, const char *fmt, ...)
144 {
145  va_list args;
146  va_start(args, fmt);
147 
148  msg_vappend(builder, node, fmt, args);
149 
150  va_end(args);
151 }
152 
153 #define CHECK_BUILDER(bld) \
154  CTASSERT((bld).event != NULL); \
155  CTASSERT((bld).arena != NULL);
156 
157 STA_DECL
158 void msg_vappend(event_builder_t builder, const node_t *node, const char *fmt, va_list args)
159 {
160  CHECK_BUILDER(builder);
161  CTASSERT(fmt != NULL);
162 
163  event_t *event = builder.event;
164 
165  if (event->segments == NULL)
166  {
167  event->segments = typevec_new(sizeof(segment_t), 2, builder.arena);
168  ARENA_IDENTIFY(event->segments, "segments", event, builder.arena);
169  }
170 
171  char *msg = str_vformat(builder.arena, fmt, args);
172 
173  segment_t segment = {
174  .node = *node,
175  .message = msg
176  };
177 
178  typevec_push(event->segments, &segment);
179 }
180 
181 STA_DECL
182 void msg_note(event_builder_t builder, const char *fmt, ...)
183 {
184  va_list args;
185  va_start(args, fmt);
186 
187  msg_vnote(builder, fmt, args);
188 
189  va_end(args);
190 }
191 
192 STA_DECL
193 void msg_vnote(event_builder_t builder, const char *fmt, va_list args)
194 {
195  CHECK_BUILDER(builder);
196  CTASSERT(fmt != NULL);
197 
198  char *msg = str_vformat(builder.arena, fmt, args);
199 
200  event_t *event = builder.event;
201 
202  if (event->notes == NULL)
203  {
204  event->notes = vector_new(4, builder.arena);
205  }
206 
207  vector_push(&event->notes, msg);
208 }
209 
210 static const char *const kSeverityNames[eSeverityTotal] = {
211 #define SEVERITY(id, name) [id] = (name),
212 #include "notify/notify.inc"
213 };
214 
215 STA_DECL
216 const char *severity_string(severity_t severity)
217 {
218  CT_ASSERT_RANGE(severity, 0, eSeverityTotal - 1);
219 
220  return kSeverityNames[severity];
221 }
STA_DECL void msg_note(event_builder_t builder, const char *fmt,...)
Definition: notify.c:182
STA_DECL const char * severity_string(severity_t severity)
get the name of a severity
Definition: notify.c:216
#define CHECK_BUILDER(bld)
Definition: notify.c:153
STA_DECL event_builder_t msg_vnotify(logger_t *logs, const diagnostic_t *diagnostic, const node_t *node, const char *fmt, va_list args)
Definition: notify.c:118
STA_DECL event_builder_t msg_notify(logger_t *logs, const diagnostic_t *diagnostic, const node_t *node, const char *fmt,...)
Definition: notify.c:105
STA_DECL void msg_append(event_builder_t builder, const node_t *node, const char *fmt,...)
Definition: notify.c:143
#define STA_DECL
sal2 annotation on function implementations to copy annotations from the declaration
CT_NODISCARD CT_PUREFN CT_STD_API bool set_contains(const set_t *set, const void *key)
check if a set contains a key
Definition: set.c:118
#define ARENA_IDENTIFY(ptr, name, parent, arena)
rename and reparent a pointer in a custom allocator
Definition: arena.h:409
#define ARENA_MALLOC(size, name, parent, arena)
allocate memory from a custom allocator
Definition: arena.h:392
STA_DECL void logger_reset(logger_t *logs)
reset the loggers messages
Definition: notify.c:89
STA_DECL bool logger_has_errors(const logger_t *logs, notify_rules_t rules)
check if the logger has any fatal errors
Definition: notify.c:44
STA_DECL arena_t * logger_get_arena(const logger_t *logs)
Definition: notify.c:97
severity_t
the default severity of a diagnostic
Definition: diagnostic.h:18
STA_DECL logger_t * logger_new(arena_t *arena)
create a new logger
Definition: notify.c:21
STA_DECL void msg_vnote(event_builder_t builder, const char *fmt, va_list args)
add a note to an existing message
Definition: notify.c:193
STA_DECL void msg_vappend(event_builder_t builder, const node_t *node, const char *fmt, va_list args)
append additional information to a message
Definition: notify.c:158
STA_DECL typevec_t * logger_get_events(const logger_t *logs)
get the events from the logger
Definition: notify.c:36
@ eSeverityTotal
Definition: diagnostic.h:22
#define CT_ASSERT_RANGE(value, min, max)
assert that a value is in a range inclusive bounds check
Definition: panic.h:148
#define CTASSERT(expr)
assert a condition, prints the condition as a message
Definition: panic.h:130
#define CTASSERTF(expr,...)
assert a condition with a message and optional format arguments
Definition: panic.h:116
CT_NODISCARD STA_FORMAT_STRING const char * fmt
Definition: str.h:68
CT_NODISCARD STA_FORMAT_STRING const char CT_NODISCARD CT_STD_API char * str_vformat(arena_t *arena, const char *fmt, va_list args)
format a string
Definition: str.c:110
CT_STD_API void * typevec_push(typevec_t *vec, const void *src)
push a value onto the vector
Definition: vector.c:156
CT_STD_API void typevec_reset(typevec_t *vec)
reset a vector
Definition: vector.c:217
CT_NODISCARD CT_PUREFN CT_STD_API size_t typevec_len(const typevec_t *vec)
get the length of a vector
Definition: vector.c:120
CT_STD_API void typevec_get(const typevec_t *vec, size_t index, STA_WRITES(vec->width) void *dst)
get an element from the vector
CT_NODISCARD CT_STD_API typevec_t * typevec_new(size_t width, size_t len, arena_t *arena)
create a new typed vector on the heap
Definition: vector.c:77
CT_NODISCARD CT_STD_API vector_t * vector_new(size_t size, arena_t *arena)
create a new vector with an initial capacity
Definition: vector.c:63
CT_STD_API void vector_push(vector_t **vector, void *value)
push a value onto the end of a vector
Definition: vector.c:108
an allocator object
Definition: arena.h:86
a diagnostic
Definition: diagnostic.h:27
severity_t severity
the severity of the diagnostic
Definition: diagnostic.h:29
an event builder handles adding additional information to an event
Definition: notify.h:68
arena_t * arena
allocation context
Definition: notify.h:73
event_t * event
the event to append to
Definition: notify.h:70
an event handle TODO: make this and segment_t opaque
Definition: notify.h:36
const diagnostic_t * diagnostic
the related diagnostic
Definition: notify.h:38
vector_t * notes
extra notes that this event is attached to
Definition: notify.h:52
typevec_t * segments
extra segments that this event is attached to
Definition: notify.h:48
a logging sink
Definition: notify.c:14
arena_t * arena
Definition: notify.c:15
typevec_t * messages
Definition: notify.c:17
a position in a source file
Definition: node.h:23
a set of rules for filtering notifications
Definition: notify.h:78
set_t * warnings_as_errors
the set of warnings to treat as errors
Definition: notify.h:81
set_t * ignored_warnings
the set of warnings to ignore this takes precedence over warnings_as_errors
Definition: notify.h:86
a segment inside an event
Definition: notify.h:57
node_t node
the related node
Definition: notify.h:59
A vector with a fixed type size.
Definition: vector.h:24