bmtoa.c revision cbc4e2be
1eaef79e5Smrg/* 2eaef79e5Smrg 3eaef79e5SmrgCopyright 1988, 1993, 1998 The Open Group 4eaef79e5Smrg 5eaef79e5SmrgPermission to use, copy, modify, distribute, and sell this software and its 6eaef79e5Smrgdocumentation for any purpose is hereby granted without fee, provided that 7eaef79e5Smrgthe above copyright notice appear in all copies and that both that 8eaef79e5Smrgcopyright notice and this permission notice appear in supporting 9eaef79e5Smrgdocumentation. 10eaef79e5Smrg 11eaef79e5SmrgThe above copyright notice and this permission notice shall be included 12eaef79e5Smrgin all copies or substantial portions of the Software. 13eaef79e5Smrg 14eaef79e5SmrgTHE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS 15eaef79e5SmrgOR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF 16eaef79e5SmrgMERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. 17eaef79e5SmrgIN NO EVENT SHALL THE OPEN GROUP BE LIABLE FOR ANY CLAIM, DAMAGES OR 18eaef79e5SmrgOTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, 19eaef79e5SmrgARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR 20eaef79e5SmrgOTHER DEALINGS IN THE SOFTWARE. 21eaef79e5Smrg 22eaef79e5SmrgExcept as contained in this notice, the name of The Open Group shall 23eaef79e5Smrgnot be used in advertising or otherwise to promote the sale, use or 24eaef79e5Smrgother dealings in this Software without prior written authorization 25eaef79e5Smrgfrom The Open Group. 26eaef79e5Smrg 27eaef79e5Smrg*/ 28eaef79e5Smrg 29eaef79e5Smrg/* 30eaef79e5Smrg * bmtoa - bitmap to ascii filter 31eaef79e5Smrg * Author: Jim Fulton, MIT X Consortium 32eaef79e5Smrg */ 33eaef79e5Smrg 34eaef79e5Smrg#ifdef HAVE_CONFIG_H 35eaef79e5Smrg# include "config.h" 36eaef79e5Smrg#endif 37eaef79e5Smrg 38eaef79e5Smrg#include <stdio.h> 39eaef79e5Smrg#include <X11/Xlib.h> 40eaef79e5Smrg#include <X11/Xutil.h> 41eaef79e5Smrg#include <X11/Xos.h> 42eaef79e5Smrg 43eaef79e5Smrg#include <X11/Xmu/Drawing.h> 44eaef79e5Smrg 45eaef79e5Smrg#include <stdlib.h> 46eaef79e5Smrg#include <unistd.h> 47cbc4e2beSmrg#ifndef HAVE_MKSTEMP 48eaef79e5Smrgextern char *mktemp(); 49eaef79e5Smrg#endif 50eaef79e5Smrg 512b32c8f7Smrgstatic char *ProgramName; 52eaef79e5Smrg 53cbc4e2beSmrgstatic void print_scanline (unsigned int width, unsigned int height, 54cbc4e2beSmrg unsigned const char *data, const char *chars); 55eaef79e5Smrg 56cbc4e2beSmrgstatic void _X_NORETURN 57eaef79e5Smrgusage (void) 58eaef79e5Smrg{ 59cbc4e2beSmrg fprintf (stderr, "usage: %s [-options ...] [filename]\n\n%s\n", 60cbc4e2beSmrg ProgramName, 61cbc4e2beSmrg "where options include:\n" 62eaef79e5Smrg " -chars cc chars to use for 0 and 1 bits, respectively\n"); 63eaef79e5Smrg exit (1); 64eaef79e5Smrg} 65eaef79e5Smrg 66eaef79e5Smrgstatic char * 67eaef79e5Smrgcopy_stdin (void) 68eaef79e5Smrg{ 69eaef79e5Smrg#ifdef WIN32 70eaef79e5Smrg static char tmpfilename[] = "/temp/bmtoa.XXXXXX"; 71eaef79e5Smrg#else 72eaef79e5Smrg static char tmpfilename[] = "/tmp/bmtoa.XXXXXX"; 73eaef79e5Smrg#endif 74eaef79e5Smrg char buf[BUFSIZ]; 75cbc4e2beSmrg FILE *fp = NULL; 76eaef79e5Smrg int nread, nwritten; 77eaef79e5Smrg 78cbc4e2beSmrg#ifndef HAVE_MKSTEMP 79cbc4e2beSmrg if (mktemp (tmpfilename) != NULL) 80cbc4e2beSmrg fp = fopen (tmpfilename, "w"); 81eaef79e5Smrg#else 82eaef79e5Smrg int fd; 83cbc4e2beSmrg if ((fd = mkstemp(tmpfilename)) >= 0) 84cbc4e2beSmrg fp = fdopen(fd, "w"); 85cbc4e2beSmrg#endif 86cbc4e2beSmrg if (fp == NULL) { 87eaef79e5Smrg fprintf (stderr, 88cbc4e2beSmrg "%s: unable to generate temporary file for stdin.\n", 89eaef79e5Smrg ProgramName); 90eaef79e5Smrg exit (1); 91eaef79e5Smrg } 92eaef79e5Smrg while (1) { 93eaef79e5Smrg buf[0] = '\0'; 94eaef79e5Smrg nread = fread (buf, 1, sizeof buf, stdin); 95eaef79e5Smrg if (nread <= 0) break; 96eaef79e5Smrg nwritten = fwrite (buf, 1, nread, fp); 97eaef79e5Smrg if (nwritten != nread) { 98eaef79e5Smrg fprintf (stderr, 99eaef79e5Smrg "%s: error copying stdin to file (%d of %d chars)\n", 100eaef79e5Smrg ProgramName, nwritten, nread); 101eaef79e5Smrg (void) fclose (fp); 102eaef79e5Smrg (void) unlink (tmpfilename); 103eaef79e5Smrg exit (1); 104eaef79e5Smrg } 105eaef79e5Smrg } 106eaef79e5Smrg (void) fclose (fp); 107eaef79e5Smrg return tmpfilename; 108eaef79e5Smrg} 109eaef79e5Smrg 110eaef79e5Smrgint 111cbc4e2beSmrgmain (int argc, char *argv[]) 112eaef79e5Smrg{ 113cbc4e2beSmrg const char *filename = NULL; 114eaef79e5Smrg int isstdin = 0; 115cbc4e2beSmrg const char *chars = "-#"; 116eaef79e5Smrg int i; 117eaef79e5Smrg unsigned int width, height; 118eaef79e5Smrg unsigned char *data; 119eaef79e5Smrg int x_hot, y_hot; 120eaef79e5Smrg int status; 121eaef79e5Smrg 122eaef79e5Smrg ProgramName = argv[0]; 123eaef79e5Smrg 124eaef79e5Smrg for (i = 1; i < argc; i++) { 125cbc4e2beSmrg const char *arg = argv[i]; 126eaef79e5Smrg 127eaef79e5Smrg if (arg[0] == '-') { 128eaef79e5Smrg switch (arg[1]) { 129eaef79e5Smrg case '\0': 130eaef79e5Smrg filename = NULL; 131eaef79e5Smrg continue; 132eaef79e5Smrg case 'c': 133eaef79e5Smrg if (++i >= argc) usage (); 134eaef79e5Smrg chars = argv[i]; 135eaef79e5Smrg continue; 136eaef79e5Smrg default: 137eaef79e5Smrg usage (); 138eaef79e5Smrg } 139eaef79e5Smrg } else { 140eaef79e5Smrg filename = arg; 141eaef79e5Smrg } 142eaef79e5Smrg } 143eaef79e5Smrg 144eaef79e5Smrg if (strlen (chars) != 2) { 145eaef79e5Smrg fprintf (stderr, 146eaef79e5Smrg "%s: bad character list \"%s\", must have exactly 2 characters\n", 147eaef79e5Smrg ProgramName, chars); 148eaef79e5Smrg exit (1); 149eaef79e5Smrg } 150eaef79e5Smrg 151eaef79e5Smrg if (!filename) { 152eaef79e5Smrg filename = copy_stdin (); 153eaef79e5Smrg isstdin = 1; 154eaef79e5Smrg } 155eaef79e5Smrg 156eaef79e5Smrg status = XmuReadBitmapDataFromFile (filename, &width, &height, &data, 157eaef79e5Smrg &x_hot, &y_hot); 158eaef79e5Smrg if (isstdin) (void) unlink (filename); /* don't need it anymore */ 159eaef79e5Smrg if (status != BitmapSuccess) { 160eaef79e5Smrg fprintf (stderr, "%s: unable to read bitmap from file \"%s\"\n", 161eaef79e5Smrg ProgramName, isstdin ? "(stdin)" : filename); 162eaef79e5Smrg exit (1); 163eaef79e5Smrg } 164eaef79e5Smrg 165eaef79e5Smrg print_scanline (width, height, data, chars); 166eaef79e5Smrg exit (0); 167eaef79e5Smrg} 168eaef79e5Smrg 169eaef79e5Smrgstatic void 170cbc4e2beSmrgprint_scanline (unsigned int width, 171cbc4e2beSmrg unsigned int height, 172cbc4e2beSmrg unsigned const char *data, 173cbc4e2beSmrg const char *chars) 174eaef79e5Smrg{ 175cbc4e2beSmrg unsigned const char *dp = data; 176eaef79e5Smrg int row, column; 177cbc4e2beSmrg static unsigned const char masktable[] = { 178eaef79e5Smrg 0x01, 0x02, 0x04, 0x08, 0x10, 0x20, 0x40, 0x80 }; 179eaef79e5Smrg int padded = ((width & 7) != 0); 180eaef79e5Smrg 181eaef79e5Smrg for (row = 0; row < height; row++) { 182eaef79e5Smrg for (column = 0; column < width; column++) { 183eaef79e5Smrg int i = (column & 7); 184eaef79e5Smrg 185eaef79e5Smrg if (*dp & masktable[i]) { 186eaef79e5Smrg putchar (chars[1]); 187eaef79e5Smrg } else { 188eaef79e5Smrg putchar (chars[0]); 189eaef79e5Smrg } 190eaef79e5Smrg 191eaef79e5Smrg if (i == 7) dp++; 192eaef79e5Smrg } 193eaef79e5Smrg putchar ('\n'); 194eaef79e5Smrg if (padded) dp++; 195eaef79e5Smrg } 196eaef79e5Smrg return; 197eaef79e5Smrg} 198eaef79e5Smrg 199