]> git.ipfire.org Git - thirdparty/binutils-gdb.git/blame - sim/common/hw-events.c
GDB copyright headers update after running GDB's copyright.py script.
[thirdparty/binutils-gdb.git] / sim / common / hw-events.c
CommitLineData
c906108c 1/* Hardware event manager.
618f726f 2 Copyright (C) 1998-2016 Free Software Foundation, Inc.
c906108c
SS
3 Contributed by Cygnus Support.
4
5This file is part of GDB, the GNU debugger.
6
7This program is free software; you can redistribute it and/or modify
8it under the terms of the GNU General Public License as published by
4744ac1b
JB
9the Free Software Foundation; either version 3 of the License, or
10(at your option) any later version.
c906108c
SS
11
12This program is distributed in the hope that it will be useful,
13but WITHOUT ANY WARRANTY; without even the implied warranty of
14MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15GNU General Public License for more details.
16
4744ac1b
JB
17You should have received a copy of the GNU General Public License
18along with this program. If not, see <http://www.gnu.org/licenses/>. */
c906108c
SS
19
20
21#include "hw-main.h"
22#include "hw-base.h"
23
24#include "sim-events.h"
25
26
27/* The hw-events object is implemented using sim-events */
28
12c4cbd5
MF
29struct hw_event
30{
c906108c
SS
31 void *data;
32 struct hw *me;
33 hw_event_callback *callback;
34 sim_event *real;
35 struct hw_event_data *entry;
36};
37
12c4cbd5
MF
38struct hw_event_data
39{
c906108c
SS
40 struct hw_event event;
41 struct hw_event_data *next;
42};
43
44void
45create_hw_event_data (struct hw *me)
46{
47 if (me->events_of_hw != NULL)
48 hw_abort (me, "stray events");
49 /* NOP */
50}
51
52void
53delete_hw_event_data (struct hw *me)
54{
4e73b9c1
SC
55 /* Remove the scheduled event. */
56 while (me->events_of_hw)
57 hw_event_queue_deschedule (me, &me->events_of_hw->event);
c906108c
SS
58}
59
60
61/* Pass the H/W event onto the real callback */
62
63static void
64bounce_hw_event (SIM_DESC sd,
65 void *data)
66{
67 /* save the data */
68 struct hw_event_data *entry = (struct hw_event_data *) data;
69 struct hw *me = entry->event.me;
70 void *event_data = entry->event.data;
71 hw_event_callback *callback = entry->event.callback;
72 struct hw_event_data **prev = &me->events_of_hw;
73 while ((*prev) != entry)
74 prev = &(*prev)->next;
75 (*prev) = entry->next;
76 hw_free (me, entry);
77 callback (me, event_data); /* may not return */
78}
79
80
81
82/* Map onto the event functions */
83
84struct hw_event *
85hw_event_queue_schedule (struct hw *me,
86 signed64 delta_time,
87 hw_event_callback *callback,
88 void *data)
89{
90 struct hw_event *event;
91 va_list dummy;
27961795 92 memset (&dummy, 0, sizeof dummy);
c906108c
SS
93 event = hw_event_queue_schedule_vtracef (me, delta_time, callback, data,
94 NULL, dummy);
95 return event;
96}
97
98struct hw_event *
99hw_event_queue_schedule_tracef (struct hw *me,
100 signed64 delta_time,
101 hw_event_callback *callback,
102 void *data,
103 const char *fmt,
104 ...)
105{
106 struct hw_event *event;
107 va_list ap;
108 va_start (ap, fmt);
109 event = hw_event_queue_schedule_vtracef (me, delta_time, callback, data, fmt, ap);
110 va_end (ap);
111 return event;
112}
113
114struct hw_event *
115hw_event_queue_schedule_vtracef (struct hw *me,
116 signed64 delta_time,
117 hw_event_callback *callback,
118 void *data,
119 const char *fmt,
120 va_list ap)
121{
122 struct hw_event_data *entry = HW_ZALLOC (me, struct hw_event_data);
123 entry->next = me->events_of_hw;
124 me->events_of_hw = entry;
125 /* fill it in */
126 entry->event.entry = entry;
127 entry->event.data = data;
128 entry->event.callback = callback;
129 entry->event.me = me;
130 entry->event.real = sim_events_schedule_vtracef (hw_system (me),
131 delta_time,
132 bounce_hw_event,
133 entry,
134 fmt, ap);
135 return &entry->event;
136}
137
138
139void
140hw_event_queue_deschedule (struct hw *me,
141 struct hw_event *event_to_remove)
142{
143/* ZAP the event but only if it is still in the event queue. Note
144 that event_to_remove is only de-referenced after its validity has
145 been confirmed. */
146 struct hw_event_data **prev;
147 for (prev = &me->events_of_hw;
148 (*prev) != NULL;
149 prev = &(*prev)->next)
150 {
151 struct hw_event_data *entry = (*prev);
152 if (&entry->event == event_to_remove)
153 {
154 sim_events_deschedule (hw_system (me),
155 entry->event.real);
156 (*prev) = entry->next;
157 hw_free (me, entry);
158 return;
159 }
160 }
161}
162
163
164signed64
165hw_event_queue_time (struct hw *me)
166{
167 return sim_events_time (hw_system (me));
168}
169
38e64f35
AC
170/* Returns the time that remains before the event is raised. */
171signed64
172hw_event_remain_time (struct hw *me, struct hw_event *event)
173{
174 signed64 t;
175
176 t = sim_events_remain_time (hw_system (me), event->real);
177 return t;
178}
c906108c
SS
179
180/* Only worry about this compling on ANSI systems.
181 Build with `make test-hw-events' in sim/<cpu> directory*/
182
183#if defined (MAIN)
184#include "sim-main.h"
185#include <string.h>
186#include <stdio.h>
187
188static void
189test_handler (struct hw *me,
190 void *data)
191{
192 int *n = data;
193 if (*n != hw_event_queue_time (me))
194 abort ();
195 *n = -(*n);
196}
197
198int
199main (int argc,
200 char **argv)
201{
202 host_callback *cb = ZALLOC (host_callback);
203 struct sim_state *sd = sim_state_alloc (0, cb);
204 struct hw *me = ZALLOC (struct hw);
205 sim_pre_argv_init (sd, "test-hw-events");
206 sim_post_argv_init (sd);
207 me->system_of_hw = sd;
208
209 printf ("Create hw-event-data\n");
210 {
211 create_hw_alloc_data (me);
212 create_hw_event_data (me);
213 delete_hw_event_data (me);
214 delete_hw_alloc_data (me);
215 }
216
217 printf ("Create hw-events\n");
218 {
219 struct hw_event *a;
220 struct hw_event *b;
221 struct hw_event *c;
222 struct hw_event *d;
223 create_hw_alloc_data (me);
224 create_hw_event_data (me);
225 a = hw_event_queue_schedule (me, 0, NULL, NULL);
226 b = hw_event_queue_schedule (me, 1, NULL, NULL);
227 c = hw_event_queue_schedule (me, 2, NULL, NULL);
228 d = hw_event_queue_schedule (me, 1, NULL, NULL);
229 hw_event_queue_deschedule (me, c);
230 hw_event_queue_deschedule (me, b);
231 hw_event_queue_deschedule (me, a);
232 hw_event_queue_deschedule (me, d);
233 c = HW_ZALLOC (me, struct hw_event);
234 hw_event_queue_deschedule (me, b); /* OOPS! */
235 hw_free (me, c);
236 delete_hw_event_data (me);
237 delete_hw_alloc_data (me);
238 }
239
240 printf ("Schedule hw-events\n");
241 {
242 struct hw_event **e;
243 int *n;
244 int i;
245 int nr = 4;
246 e = HW_NZALLOC (me, struct hw_event *, nr);
247 n = HW_NZALLOC (me, int, nr);
248 create_hw_alloc_data (me);
249 create_hw_event_data (me);
250 for (i = 0; i < nr; i++)
251 {
252 n[i] = i;
253 e[i] = hw_event_queue_schedule (me, i, test_handler, &n[i]);
254 }
255 sim_events_preprocess (sd, 1, 1);
256 for (i = 0; i < nr; i++)
257 {
258 if (sim_events_tick (sd))
259 sim_events_process (sd);
260 }
261 for (i = 0; i < nr; i++)
262 {
263 if (n[i] != -i)
264 abort ();
265 hw_event_queue_deschedule (me, e[i]);
266 }
267 hw_free (me, n);
268 hw_free (me, e);
269 delete_hw_event_data (me);
270 delete_hw_alloc_data (me);
271 }
272
273 return 0;
274}
275#endif