13 enum { AS_GNU, AS_NASM };
15 int csprite(struct image *img, int x, int y, int xsz, int ysz);
16 int proc_sheet(const char *fname);
17 void print_usage(const char *argv0);
19 int tile_xsz, tile_ysz;
23 int fbpitch, fbwidth = 320;
24 const char *name = "sprite";
28 const char *wrop = "mov";
30 int main(int argc, char **argv)
35 for(i=1; i<argc; i++) {
36 if(argv[i][0] == '-') {
37 if(strcmp(argv[i], "-s") == 0 || strcmp(argv[i], "-size") == 0) {
38 if(sscanf(argv[++i], "%dx%d", &tile_xsz, &tile_ysz) != 2 ||
39 tile_xsz <= 0 || tile_ysz <= 0) {
40 fprintf(stderr, "%s must be followed by WIDTHxHEIGHT\n", argv[i - 1]);
44 } else if(strcmp(argv[i], "-r") == 0 || strcmp(argv[i], "-rect") == 0) {
46 if(sscanf(argv[++i], "%dx%d+%d+%d", &rect.w, &rect.h, &rect.x, &rect.y) < 2 || rect.w <= 0 || rect.h <= 0) {
47 fprintf(stderr, "%s must be followed by WIDTHxHEIGHT+X+Y\n", argv[i - 1]);
51 } else if(strcmp(argv[i], "-p") == 0 || strcmp(argv[i], "-pad") == 0) {
52 padding = strtol(argv[++i], &endp, 10);
53 if(endp == argv[i] || padding < 0) {
54 fprintf(stderr, "%s must be followed by a positive number\n", argv[i - 1]);
58 } else if(strcmp(argv[i], "-coffset") == 0) {
59 cmap_offs = strtol(argv[++i], &endp, 10);
60 if(endp == argv[i] || cmap_offs < 0 || cmap_offs >= 256) {
61 fprintf(stderr, "-coffset must be followed by a valid colormap offset\n");
65 } else if(strcmp(argv[i], "-fbwidth") == 0) {
66 fbwidth = atoi(argv[++i]);
68 fprintf(stderr, "-fbwidth must be followed by a positive number\n");
72 } else if(strcmp(argv[i], "-fbpitch") == 0) {
73 fbpitch = atoi(argv[++i]);
75 fprintf(stderr, "-fbpitch must be followed by a positive number\n");
79 } else if(strcmp(argv[i], "-n") == 0 || strcmp(argv[i], "-name") == 0) {
82 fprintf(stderr, "%s must be followed by a name\n", argv[i - 1]);
86 } else if(strcmp(argv[i], "-k") == 0 || strcmp(argv[i], "-key") == 0) {
87 ckey = strtol(argv[++i], &endp, 10);
88 if(endp == argv[i] || ckey < 0) {
89 fprintf(stderr, "%s must be followed by a valid color key\n", argv[i - 1]);
93 } else if(strcmp(argv[i], "-conv565") == 0) {
96 } else if(strcmp(argv[i], "-x") == 0 || strcmp(argv[i], "-xor") == 0) {
99 } else if(strcmp(argv[i], "-gas") == 0) {
102 } else if(strcmp(argv[i], "-nasm") == 0) {
105 } else if(strcmp(argv[i], "-h") == 0 || strcmp(argv[i], "-help") == 0) {
106 print_usage(argv[0]);
110 fprintf(stderr, "invalid option: %s\n", argv[i]);
111 print_usage(argv[0]);
116 if(proc_sheet(argv[i]) == -1) {
125 /* prototype of generated function is (void *fb, int x, int y, int idx) */
126 const char *prefixfmt[] = {
127 /* GNU assembler template */
133 "\tmov 12(%%esp), %%eax\n"
136 "\tadd 8(%%esp), %%eax\n"
137 "\tadd 4(%%esp), %%eax\n"
138 "\tmov %%eax, %%edx\n"
139 "\tmov 16(%%esp), %%eax\n"
140 "\tjmp *tiletab(,%%eax,4)\n\n"
144 /* TODO hardcoding the 16bpp changes for now, generalize later
145 * and while we're at it, let's get rid of the mul too ...
147 "\tsection .text USE32\n"
153 "\tmov eax, [esp + 12]\n"
160 "\tadd eax, [esp + 8]\n"
161 "\tadd eax, [esp + 8]\n"
162 "\tadd eax, [esp + 4]\n"
164 "\tmov eax, [esp + 16]\n"
165 "\tjmp [titletab + eax * 4]\n\n"
176 "\tjmp [titletab + ecx * 4]\n\n"
180 int proc_sheet(const char *fname)
182 int i, j, x, y, num_xtiles, num_ytiles, xsz, ysz, tidx;
185 if(load_image(&img, fname) == -1) {
186 fprintf(stderr, "failed to load image: %s\n", fname);
191 if(conv_image_rgb565(&tmp, &img) == -1) {
192 fprintf(stderr, "failed to convert image to 16bpp 565: %s\n", fname);
200 if(!fbpitch) fbpitch = fbwidth * img.bpp / 8;
208 num_xtiles = num_ytiles = 1;
209 xsz = rect.w - padding;
210 ysz = rect.h - padding;
213 num_xtiles = num_ytiles = 0;
217 i += tile_xsz + padding;
222 i += tile_ysz + padding;
225 num_xtiles = rect.w / tile_xsz;
226 num_ytiles = rect.h / tile_ysz;
232 printf(prefixfmt[asyntax], name, name, name, name, name, fbpitch, name);
233 for(i=0; i<num_ytiles*num_xtiles; i++) {
234 if(asyntax == AS_GNU) {
235 printf("\t.long tile%d\n", i);
237 printf("\tdd tile%d\n", i);
244 for(i=0; i<num_ytiles; i++) {
246 for(j=0; j<num_xtiles; j++) {
247 printf("tile%d:\n", tidx++);
248 csprite(&img, x, y, xsz, ysz);
266 unsigned char op, val;
270 int csprite(struct image *img, int x, int y, int xsz, int ysz)
272 int i, j, numops, mode, new_mode, start, skip_acc, lenbytes, pixsz = img->bpp / 8;
273 unsigned char *pptr = img->pixels + y * img->scansz + x * pixsz;
274 struct csop *ops, *optr;
276 ops = optr = alloca((xsz + 1) * ysz * sizeof *ops);
278 for(i=0; i<ysz; i++) {
283 optr++->op = CSOP_ENDL;
286 for(j=0; j<xsz; j++) {
287 if(memcmp(pptr, &ckey, pixsz) == 0) {
288 new_mode = CSOP_SKIP;
290 new_mode = CSOP_COPY;
293 if(new_mode != mode) {
297 optr->len = j - start;
305 pptr += img->scansz - xsz * pixsz;
310 optr->len = xsz - start;
316 pptr = img->pixels + y * img->scansz + x * img->bpp / 8;
319 /* edx points to dest */
320 for(i=0; i<numops; i++) {
323 skip_acc += optr->len;
324 pptr += optr->len * pixsz;
328 skip_acc += fbwidth - xsz;
329 pptr += img->scansz - xsz * pixsz;
334 if(asyntax == AS_GNU) {
335 printf("\tadd $%d, %%edx\n", skip_acc * pixsz);
337 printf("\tadd edx, %d\n", skip_acc * pixsz);
342 lenbytes = optr->len * pixsz;
343 for(j=0; j<lenbytes / 4; j++) {
344 if(asyntax == AS_GNU) {
345 printf("\t%sl $0x%x, %d(%%edx)\n", wrop, *(uint32_t*)pptr, j * 4);
347 printf("\t%s dword [edx + %d], 0x%x\n", wrop, j * 4, *(uint32_t*)pptr);
352 switch(lenbytes % 4) {
354 if(asyntax == AS_GNU) {
355 printf("\t%sb $0x%x, %d(%%edx)\n", wrop, (unsigned int)*pptr++, j++);
357 printf("\t%s byte [edx + %d], 0x%x\n", wrop, j++, (unsigned int)*pptr++);
360 if(asyntax == AS_GNU) {
361 printf("\t%sw $0x%x, %d(%%edx)\n", wrop, (unsigned int)*(uint16_t*)pptr, j);
363 printf("\t%s word [edx + %d], 0x%x\n", wrop, j, (unsigned int)*(uint16_t*)pptr);
369 if(asyntax == AS_GNU) {
370 printf("\t%sb $0x%x, %d(%%edx)\n", wrop, (unsigned int)*pptr++, j++);
372 printf("\t%s byte [edx + %d], 0x%x\n", wrop, j++, (unsigned int)*pptr++);
377 skip_acc = optr->len;
381 printf("\t%c invalid op: %d\n", asyntax == AS_GNU ? '#' : ';', optr->op);
390 void print_usage(const char *argv0)
392 printf("Usage: %s [options] <spritesheet>\n", argv0);
393 printf("Options:\n");
394 printf(" -s,-size <WxH>: tile size (default: whole image)\n");
395 printf(" -r,-rect <WxH+X+Y>: use rectangle of the input image (default: whole image)\n");
396 printf(" -p,-pad <N>: how many pixels to skip between tiles in source image (default: 0)\n");
397 printf(" -coffset <offs>: colormap offset [0, 255] (default: 0)\n");
398 printf(" -fbpitch <pitch>: target framebuffer pitch (scanline size in bytes)\n");
399 printf(" -k,-key <color>: color-key for transparency (default: 0)\n");
400 printf(" -conv565: convert image to 16bpp 565 before processing\n");
401 printf(" -x,-xor: use XOR for writing pixels instead of MOV\n");
402 printf(" -gas: output GNU assembler code (default)\n");
403 printf(" -nasm: output NASM-compatible code\n");
404 printf(" -h: print usage and exit\n");