]> git.sur5r.net Git - cc65/blob - src/da65/attrtab.c
Add new SKIP range
[cc65] / src / da65 / attrtab.c
1 /*****************************************************************************/
2 /*                                                                           */
3 /*                                 attrtab.c                                 */
4 /*                                                                           */
5 /*                       Disassembler attribute table                        */
6 /*                                                                           */
7 /*                                                                           */
8 /*                                                                           */
9 /* (C) 2000-2003 Ullrich von Bassewitz                                       */
10 /*               Römerstrasse 52                                             */
11 /*               D-70794 Filderstadt                                         */
12 /* EMail:        uz@cc65.org                                                 */
13 /*                                                                           */
14 /*                                                                           */
15 /* This software is provided 'as-is', without any expressed or implied       */
16 /* warranty.  In no event will the authors be held liable for any damages    */
17 /* arising from the use of this software.                                    */
18 /*                                                                           */
19 /* Permission is granted to anyone to use this software for any purpose,     */
20 /* including commercial applications, and to alter it and redistribute it    */
21 /* freely, subject to the following restrictions:                            */
22 /*                                                                           */
23 /* 1. The origin of this software must not be misrepresented; you must not   */
24 /*    claim that you wrote the original software. If you use this software   */
25 /*    in a product, an acknowledgment in the product documentation would be  */
26 /*    appreciated but is not required.                                       */
27 /* 2. Altered source versions must be plainly marked as such, and must not   */
28 /*    be misrepresented as being the original software.                      */
29 /* 3. This notice may not be removed or altered from any source              */
30 /*    distribution.                                                          */
31 /*                                                                           */
32 /*****************************************************************************/
33
34
35
36 #include <stdio.h>
37 #include <string.h>
38
39 /* common */
40 #include "xmalloc.h"
41 #include "xsprintf.h"
42
43 /* da65 */
44 #include "code.h"
45 #include "error.h"
46 #include "global.h"
47 #include "output.h"
48 #include "attrtab.h"
49
50
51
52 /*****************************************************************************/
53 /*                                   Data                                    */
54 /*****************************************************************************/
55
56
57
58 /* Attribute table */
59 static unsigned char AttrTab [0x10000];
60
61 /* Symbol table */
62 static const char* SymTab [0x10000];
63
64
65
66 /*****************************************************************************/
67 /*                                   Code                                    */
68 /*****************************************************************************/
69
70
71
72 static void AddrCheck (unsigned Addr)
73 /* Check if the given address has a valid range */
74 {
75     if (Addr >= 0x10000) {
76         Error ("Address out of range: %08X", Addr);
77     }
78 }
79
80
81
82 void MarkRange (unsigned Start, unsigned End, attr_t Attr)
83 /* Mark a range with the given attribute */
84 {
85     /* Do it easy here... */
86     while (Start <= End) {
87         MarkAddr (Start++, Attr);
88     }
89 }
90
91
92
93 void MarkAddr (unsigned Addr, attr_t Attr)
94 /* Mark an address with an attribute */
95 {
96     /* Check the given address */
97     AddrCheck (Addr);
98
99     /* We must not have more than one style bit */
100     if (Attr & atStyleMask) {
101         if (AttrTab[Addr] & atStyleMask) {
102             Error ("Duplicate style for address %04X", Addr);
103         }
104     }
105
106     /* Set the style */
107     AttrTab[Addr] |= Attr;
108 }
109
110
111
112 const char* MakeLabelName (unsigned Addr)
113 /* Make the default label name from the given address and return it in a
114  * static buffer.
115  */
116 {
117     static char LabelBuf [32];
118     xsprintf (LabelBuf, sizeof (LabelBuf), "L%04X", Addr);
119     return LabelBuf;
120 }
121
122
123
124 void AddLabel (unsigned Addr, attr_t Attr, const char* Name)
125 /* Add a label */
126 {
127     /* Get an existing label attribute */
128     attr_t ExistingAttr = GetLabelAttr (Addr);
129
130     /* Must not have two symbols for one address */
131     if (ExistingAttr != atNoLabel) {
132         /* Allow redefinition if identical */
133         if (ExistingAttr == Attr && strcmp (SymTab[Addr], Name) == 0) {
134             return;
135         }
136         Error ("Duplicate label for address %04X: %s/%s", Addr, SymTab[Addr], Name);
137     }
138
139     /* Create a new label */
140     SymTab[Addr] = xstrdup (Name);
141
142     /* Remember the attribute */
143     AttrTab[Addr] |= Attr;
144 }
145
146
147
148 void AddExtLabelRange (unsigned Addr, const char* Name, unsigned Count)
149 /* Add an external label for a range. The first entry gets the label "Name"
150  * while the others get "Name+offs".
151  */
152 {
153     /* Define the label */
154     AddLabel (Addr, atExtLabel, Name);
155
156     /* Define dependent labels if necessary */
157     if (Count > 1) {
158         unsigned Offs;
159
160         /* Allocate memory for the dependent label names */
161         unsigned NameLen = strlen (Name);
162         char*    DepName = xmalloc (NameLen + 7);
163         char*    DepOffs = DepName + NameLen + 1;
164
165         /* Copy the original name into the buffer */
166         memcpy (DepName, Name, NameLen);
167         DepName[NameLen] = '+';
168
169         /* Define the labels */
170         for (Offs = 1; Offs < Count; ++Offs) {
171             sprintf (DepOffs, "%u", Offs);
172             AddLabel (Addr + Offs, atDepLabel, DepName);
173         }
174
175         /* Free the name buffer */
176         xfree (DepName);
177     }
178 }
179
180
181
182 int HaveLabel (unsigned Addr)
183 /* Check if there is a label for the given address */
184 {
185     /* Check the given address */
186     AddrCheck (Addr);
187
188     /* Check for a label */
189     return (SymTab[Addr] != 0);
190 }
191
192
193
194 int MustDefLabel (unsigned Addr)
195 /* Return true if we must define a label for this address, that is, if there
196  * is a label at this address, and it is an external or internal label.
197  */
198 {
199     /* Get the label attribute */
200     attr_t A = GetLabelAttr (Addr);
201
202     /* Check for an internal or external label */
203     return (A == atExtLabel || A == atIntLabel);
204 }
205
206
207
208 const char* GetLabel (unsigned Addr)
209 /* Return the label for an address */
210 {
211     /* Check the given address */
212     AddrCheck (Addr);
213
214     /* Return the label if any */
215     return SymTab[Addr];
216 }
217
218
219
220 unsigned char GetStyleAttr (unsigned Addr)
221 /* Return the style attribute for the given address */
222 {
223     /* Check the given address */
224     AddrCheck (Addr);
225
226     /* Return the attribute */
227     return (AttrTab[Addr] & atStyleMask);
228 }
229
230
231
232 unsigned char GetLabelAttr (unsigned Addr)
233 /* Return the label attribute for the given address */
234 {
235     /* Check the given address */
236     AddrCheck (Addr);
237
238     /* Return the attribute */
239     return (AttrTab[Addr] & atLabelMask);
240 }
241
242
243
244 static void DefineConst (unsigned Addr)
245 /* Define an address constant */
246 {
247     Output ("%s", SymTab [Addr]);
248     Indent (AIndent);
249     Output ("= $%04X", Addr);
250     LineFeed ();
251 }
252
253
254
255 void DefOutOfRangeLabels (void)
256 /* Output any labels that are out of the loaded code range */
257 {
258     unsigned long Addr;
259
260     SeparatorLine ();
261
262     /* Low range */
263     Addr = 0;
264     while (Addr < CodeStart) {
265         if (MustDefLabel (Addr)) {
266             DefineConst (Addr);
267         }
268         ++Addr;
269     }
270
271     /* Skip areas in code range */
272     while (Addr <= CodeEnd) {
273         if ((AttrTab[Addr] & atStyleMask) == atSkip && MustDefLabel (Addr)) {
274             DefineConst (Addr);
275         }
276         ++Addr;
277     }
278
279     /* High range */
280     while (Addr < 0x10000) {
281         if (MustDefLabel (Addr)) {
282             DefineConst (Addr);
283         }
284         ++Addr;
285     }
286
287     SeparatorLine ();
288 }
289
290
291