home *** CD-ROM | disk | FTP | other *** search
/ Peanuts NeXT Software Archives / Peanuts-3.iso / Graphics / convertors / mpnttotiff.c < prev    next >
Encoding:
C/C++ Source or Header  |  1993-07-23  |  4.0 KB  |  169 lines

  1. /* Convert Macintosh screen shot (MPNT/PNTG) to NeXT TIFF
  2.  * Eric P. Scott, San Francisco State University, October 1990
  3.  *
  4.  * This input for this is the data fork of a Mac file created
  5.  * with Command-Shift-3
  6.  *
  7.  * "For more information see"
  8.  *   Tag Image File Format Specification, Revision 5.0
  9.  *   Macintosh Technical Note #86, MacPaint Document Format
  10.  *   Macintosh Technical Note #171, _PackBits Data Format
  11.  *
  12.  * Small-screen Macs (e.g. SEs) save in the normal orientation
  13.  * Large-screen Macs (e.g. IIs) rotate their bitmap--use -r option
  14.  *
  15.  * To compile:
  16.  * cc -s -o mpnttotiff -O -bsd mpnttotiff.c -lsys_s
  17.  */
  18. #include <stdio.h>
  19. #ifndef lint
  20. static char sccsid[]="@(#)mpnttotiff  1.0  (SFSU) 10/24/90";
  21. #endif
  22.  
  23. struct {
  24.     long tiff_magic, tiff_firstifd;
  25.     short tiff_ifdents;
  26.     struct ifdent {
  27.         short ifd_tag, ifd_type;
  28.         long ifd_len, ifd_off;
  29.     } tiff_ifd[8];
  30.     long tiff_ifdchain;
  31. } header={    /* warning: nonportable */
  32.     0x4d4d002aL, 8L, 8, {
  33.         { 255, 3, 1L, 1L<<16 },        /* SubfileType */
  34.         { 256, 3, 1L, 720L<<16 },    /* ImageWidth */
  35.         { 257, 3, 1L, 576L<<16 },    /* ImageLength */
  36.         { 262, 3, 1L, 1L<<16 },        /* PhotometricInterpretation */
  37.         { 277, 3, 1L, 1L<<16 },        /* SamplesPerPixel */
  38.         { 284, 3, 1L, 2L<<16 },        /* PlanarConfiguration */
  39.         { 258, 3, 1L, 1L<<16 },        /* BitsPerSample */
  40.         { 273, 4, 1L, 110L }        /* StripOffsets */
  41.     }, 0L
  42. };
  43. int rotate=0;
  44.  
  45. main(argc, argv)
  46. int argc;
  47. char *argv[];
  48. {
  49.     extern int optind;
  50.     extern char *optarg;
  51.     register unsigned char *q, *p;
  52.     int c;
  53.     long lbuf[12960];    /* space for 576x720x1 image */
  54.  
  55.     while ((c=getopt(argc, argv, "r"))!=EOF) switch (c) {
  56.     case 'r':
  57.         rotate++;
  58.         break;
  59.     default:
  60.     usage:
  61.         (void)fprintf(stderr, "Usage: %s [-r] paint-file tiff-file\n",
  62.             *argv);
  63.         (void)fputs("\t-r  rotate 90 degrees counterclockwise\n",
  64.             stderr);
  65.         exit(1);
  66.         break;
  67.     }
  68.  
  69.     if (argc-optind!=2) goto usage;
  70.     if (strcmp(argv[optind], "-")&&!freopen(argv[optind], "r", stdin)) {
  71.         perror(argv[optind]);
  72.         exit(1);
  73.     }
  74.     (void)fread((char *)lbuf, 512, 1, stdin);
  75.     if (*lbuf!=0L) {
  76.         (void)fprintf(stderr,
  77.             "%s: input isn't in a format I can deal with\n",
  78.             *argv);
  79.         exit(1);
  80.     }
  81.  
  82.     p=(unsigned char *)&lbuf[sizeof lbuf/sizeof lbuf[0]];
  83.     q=(unsigned char *)lbuf;
  84.     bzero((char *)q, sizeof lbuf);
  85.     while ((c=getchar())!=EOF) {    /* UnPackBits */
  86.         /* I'm cheating: you're supposed to do this a row at a time */
  87.         register int b;
  88.  
  89.         if ((c&=255)<=127) {
  90.             do {
  91.                 if (q>=p) {
  92.                 ovfl:
  93.                     (void)fprintf(stderr,
  94. "%s: internal buffer overflow\n", *argv);
  95.                     exit(1);
  96.                 }
  97.                 if ((b=getchar())==EOF) {
  98.                 tooshort:
  99.                     (void)fprintf(stderr,
  100. "%s: EOF in sequence\n", *argv);
  101.                     exit(1);
  102.                 }
  103.                 *q++=b;
  104.             } while (--c>=0);
  105.         }
  106.         else {
  107.             c-=256;
  108.             if ((b=getchar())==EOF) goto tooshort;
  109.             do {
  110.                 if (q>=p) goto ovfl;
  111.                 *q++=b;
  112.             } while (++c<=0);
  113.         }
  114.     }
  115.  
  116.     if (strcmp(argv[++optind], "-")&&!freopen(argv[optind], "w", stdout)) {
  117.         perror(argv[optind]);
  118.         exit(1);
  119.     }
  120.     if (!rotate) {    /* swap dimensions */
  121.         header.tiff_ifd[1].ifd_off=576L<<16;
  122.         header.tiff_ifd[2].ifd_off=720L<<16;
  123.     }
  124.     (void)fwrite((char *)&header, sizeof header, 1, stdout);
  125.  
  126.     if (rotate) {    /* rotate 90 degrees clockwise and invert */
  127.         long obuf[180];
  128.  
  129.         p=(unsigned char *)&lbuf[12816];
  130.         c=72; do {
  131.             register int r;
  132.  
  133.             q=(unsigned char *)obuf;
  134.             bzero((char *)q, sizeof obuf);
  135.             r=90; do {
  136. #define twist(m) \
  137.     if ((*p&m)==0) *q=1; \
  138.     if ((p[72]&m)==0) *q|=2; \
  139.     if ((p[144]&m)==0) *q|=4; \
  140.     if ((p[216]&m)==0) *q|=8; \
  141.     if ((p[288]&m)==0) *q|=16; \
  142.     if ((p[360]&m)==0) *q|=32; \
  143.     if ((p[432]&m)==0) *q|=64; \
  144.     if ((p[504]&m)==0) *q|=128
  145.                 twist(128); q+=90;
  146.                 twist(64); q+=90;
  147.                 twist(32); q+=90;
  148.                 twist(16); q+=90;
  149.                 twist(8); q+=90;
  150.                 twist(4); q+=90;
  151.                 twist(2); q+=90;
  152.                 twist(1); q-=629;
  153.                 p-=576;
  154.             } while (--r>0);
  155.             (void)fwrite((char *)obuf, sizeof obuf, 1, stdout);
  156.             p+=51841;    /* 1+sizeof lbuf */
  157.         } while (--c>0);
  158.     }
  159.     else {    /* just invert */
  160.         p=(unsigned char *)&lbuf[sizeof lbuf/sizeof lbuf[0]];
  161.         q=(unsigned char *)lbuf;
  162.         do {
  163.             putchar(*q++^255);
  164.         } while (q<p);
  165.     }
  166.     (void)fflush(stdout);
  167.     exit(0);
  168. }
  169.