1 /*****************************************************************************/
5 /* Assembler output for the sp65 sprite and bitmap utility */
9 /* (C) 2012, Ullrich von Bassewitz */
10 /* Roemerstrasse 52 */
11 /* D-70794 Filderstadt */
12 /* EMail: uz@cc65.org */
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. */
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: */
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 */
32 /*****************************************************************************/
53 /*****************************************************************************/
55 /*****************************************************************************/
59 static int ValidIdentifier (const char* L)
60 /* Check an assembler label for validity */
62 /* Must begin with underscore or alphabetic character */
63 if (*L != '_' && !IsAlpha (*L)) {
68 /* Remainder must be as above plus digits */
70 if (*L != '_' && !IsAlNum (*L)) {
82 static unsigned GetBytesPerLine (const Collection* A)
83 /* Return the number of bytes per line from the attribute collection A */
86 unsigned BytesPerLine = 16;
88 /* Check for a bytesperline attribute */
89 const char* V = GetAttrVal (A, "bytesperline");
90 if ((V && sscanf (V, "%u%c", &BytesPerLine, &C) != 1) ||
91 (BytesPerLine < 1 || BytesPerLine > 64)) {
92 Error ("Invalid value for attribute `bytesperline'");
99 static unsigned GetBase (const Collection* A)
100 /* Return the number base from the attribute collection A */
105 /* Check for a base attribute */
106 const char* V = GetAttrVal (A, "base");
107 if ((V && sscanf (V, "%u%c", &Base, &C) != 1) ||
108 (Base != 2 && Base != 10 && Base != 16)) {
109 Error ("Invalid value for attribute `base'");
116 static const char* GetIdentifier (const Collection* A)
117 /* Return the label identifier from the attribute collection A */
119 /* Check for a ident attribute */
120 const char* Ident = GetAttrVal (A, "ident");
121 if (Ident && !ValidIdentifier (Ident)) {
122 Error ("Invalid value for attribute `ident'");
129 void WriteAsmFile (const StrBuf* Data, const Collection* A, const Bitmap* B)
130 /* Write the contents of Data to the given file in assembler (ca65) format */
137 /* Get the name of the image */
138 const StrBuf* S = GetBitmapName (B);
140 /* Get the file name */
141 const char* Name = NeedAttrVal (A, "name", "write");
143 /* Check the number of bytes per line */
144 unsigned BytesPerLine = GetBytesPerLine (A);
146 /* Get the number base */
147 unsigned Base = GetBase (A);
149 /* Get the identifier */
150 const char* Ident = GetIdentifier (A);
152 /* Open the output file */
153 F = fopen (Name, "w");
155 Error ("Cannot open output file `%s': %s", Name, strerror (errno));
158 /* Write a readable header */
161 "; This file was generated by %s %s from\n"
162 "; %.*s (%ux%u, %u colors%s)\n"
166 GetVersionAsString (),
167 SB_GetLen (S), SB_GetConstBuf (S),
168 GetBitmapWidth (B), GetBitmapHeight (B),
170 BitmapIsIndexed (B)? ", indexed" : "");
172 /* If we have an assembler label, output that */
182 GetBitmapHeight (B));
186 D = SB_GetConstBuf (Data);
187 Size = SB_GetLen (Data);
192 /* Output one line */
193 unsigned Chunk = Size;
194 if (Chunk > BytesPerLine) {
195 Chunk = BytesPerLine;
197 fputs (" .byte ", F);
198 for (I = 0; I < Chunk; ++I) {
199 unsigned char V = *D++;
205 fprintf (F, "%%%u%u%u%u%u%u%u%u",
206 (V >> 7) & 0x01, (V >> 6) & 0x01,
207 (V >> 5) & 0x01, (V >> 4) & 0x01,
208 (V >> 3) & 0x01, (V >> 2) & 0x01,
209 (V >> 1) & 0x01, (V >> 0) & 0x01);
213 fprintf (F, "%u", V);
217 fprintf (F, "$%02X", V);
224 /* Bump the counters */
228 /* Terminate the .proc if we had an identifier */
230 fputs (".endproc\n", F);
233 /* Add an empty line at the end */
237 if (fclose (F) != 0) {
238 Error ("Error closing output file `%s': %s", Name, strerror (errno));