2 Copyright (C) 1998-2023 Free Software Foundation, Inc.
3 Contributed by Andrew Cagney and Cygnus Solutions.
5 This file is part of GDB, the GNU debugger.
7 This program is free software; you can redistribute it and/or modify
8 it under the terms of the GNU General Public License as published by
9 the Free Software Foundation; either version 3 of the License, or
10 (at your option) any later version.
12 This program is distributed in the hope that it will be useful,
13 but WITHOUT ANY WARRANTY; without even the implied warranty of
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 GNU General Public License for more details.
17 You should have received a copy of the GNU General Public License
18 along with this program. If not, see <http://www.gnu.org/licenses/>. */
20 /* This must come before any other includes. */
35 struct hw_port_edge
*next
;
36 object_disposition disposition
;
41 hw_port_event_method
*to_port_event
;
42 const struct hw_port_descriptor
*ports
;
43 struct hw_port_edge
*edges
;
46 const struct hw_port_descriptor empty_hw_ports
[] =
52 panic_hw_port_event (struct hw
*me
,
58 hw_abort (me
, "no port method");
62 create_hw_port_data (struct hw
*me
)
64 me
->ports_of_hw
= HW_ZALLOC (me
, struct hw_port_data
);
65 set_hw_port_event (me
, panic_hw_port_event
);
66 set_hw_ports (me
, empty_hw_ports
);
70 delete_hw_port_data (struct hw
*me
)
72 hw_free (me
, me
->ports_of_hw
);
73 me
->ports_of_hw
= NULL
;
77 set_hw_ports (struct hw
*me
,
78 const struct hw_port_descriptor ports
[])
80 me
->ports_of_hw
->ports
= ports
;
84 set_hw_port_event (struct hw
*me
,
85 hw_port_event_method
*port_event
)
87 me
->ports_of_hw
->to_port_event
= port_event
;
92 attach_hw_port_edge (struct hw
*me
,
93 struct hw_port_edge
**list
,
97 object_disposition disposition
)
99 struct hw_port_edge
*new_edge
= HW_ZALLOC (me
, struct hw_port_edge
);
100 new_edge
->my_port
= my_port
;
101 new_edge
->dest
= dest
;
102 new_edge
->dest_port
= dest_port
;
103 new_edge
->next
= *list
;
104 new_edge
->disposition
= disposition
;
110 detach_hw_port_edge (struct hw
*me
,
111 struct hw_port_edge
**list
,
116 while (*list
!= NULL
)
118 struct hw_port_edge
*old_edge
= *list
;
119 if (old_edge
->dest
== dest
120 && old_edge
->dest_port
== dest_port
121 && old_edge
->my_port
== my_port
)
123 if (old_edge
->disposition
== permenant_object
)
124 hw_abort (me
, "attempt to delete permenant port edge");
125 *list
= old_edge
->next
;
126 hw_free (me
, old_edge
);
130 hw_abort (me
, "attempt to delete unattached port");
136 clean_hw_port_edges (struct hw_port_edge
**list
)
138 while (*list
!= NULL
)
140 struct hw_port_edge
*old_edge
= *list
;
141 switch (old_edge
->disposition
)
143 case permenant_object
:
144 list
= &old_edge
->next
;
146 case temporary_object
:
147 *list
= old_edge
->next
;
148 hw_free (me
, old_edge
);
159 hw_port_event (struct hw
*me
,
163 int found_an_edge
= 0;
164 struct hw_port_edge
*edge
;
165 /* device's lines directly connected */
166 for (edge
= me
->ports_of_hw
->edges
;
170 if (edge
->my_port
== my_port
)
172 edge
->dest
->ports_of_hw
->to_port_event (edge
->dest
,
181 hw_abort (me
, "No edge for port %d", my_port
);
186 hw_port_attach (struct hw
*me
,
190 object_disposition disposition
)
192 attach_hw_port_edge (me
,
193 &me
->ports_of_hw
->edges
,
202 hw_port_detach (struct hw
*me
,
207 detach_hw_port_edge (me
,
208 &me
->ports_of_hw
->edges
,
216 hw_port_traverse (struct hw
*me
,
217 hw_port_traverse_function
*handler
,
220 struct hw_port_edge
*port_edge
;
221 for (port_edge
= me
->ports_of_hw
->edges
;
223 port_edge
= port_edge
->next
)
225 handler (me
, port_edge
->my_port
,
226 port_edge
->dest
, port_edge
->dest_port
,
233 hw_port_decode (struct hw
*me
,
234 const char *port_name
,
235 port_direction direction
)
237 if (port_name
== NULL
|| port_name
[0] == '\0')
239 if (isdigit (port_name
[0]))
241 return strtoul (port_name
, NULL
, 0);
245 const struct hw_port_descriptor
*ports
=
246 me
->ports_of_hw
->ports
;
249 while (ports
->name
!= NULL
)
251 if (ports
->direction
== bidirect_port
252 || ports
->direction
== direction
)
254 if (ports
->nr_ports
> 0)
256 int len
= strlen (ports
->name
);
257 if (strncmp (port_name
, ports
->name
, len
) == 0)
259 if (port_name
[len
] == '\0')
260 return ports
->number
;
261 else if (isdigit (port_name
[len
]))
263 int port
= (ports
->number
264 + strtoul (&port_name
[len
], NULL
, 0));
265 if (port
>= ports
->number
+ ports
->nr_ports
)
267 "Port %s out of range",
273 else if (strcmp (port_name
, ports
->name
) == 0)
274 return ports
->number
;
280 hw_abort (me
, "Unrecognized port %s", port_name
);
286 hw_port_encode (struct hw
*me
,
290 port_direction direction
)
292 const struct hw_port_descriptor
*ports
= NULL
;
293 ports
= me
->ports_of_hw
->ports
;
295 while (ports
->name
!= NULL
)
297 if (ports
->direction
== bidirect_port
298 || ports
->direction
== direction
)
300 if (ports
->nr_ports
> 0)
302 if (port_number
>= ports
->number
303 && port_number
< ports
->number
+ ports
->nr_ports
)
305 strcpy (buf
, ports
->name
);
306 sprintf (buf
+ strlen (buf
), "%d", port_number
- ports
->number
);
307 if (strlen (buf
) >= sizeof_buf
)
308 hw_abort (me
, "hw_port_encode: buffer overflow");
314 if (ports
->number
== port_number
)
316 if (strlen (ports
->name
) >= sizeof_buf
)
317 hw_abort (me
, "hw_port_encode: buffer overflow");
318 strcpy (buf
, ports
->name
);
326 sprintf (buf
, "%d", port_number
);
327 if (strlen (buf
) >= sizeof_buf
)
328 hw_abort (me
, "hw_port_encode: buffer overflow");