Blame converter/pbm/pbmtoppa/pbm.c

Packit 78deda
/* pbm.c
Packit 78deda
 * code for reading the header of an ASCII PBM file
Packit 78deda
 * Copyright (c) 1998 Tim Norman.  See LICENSE for details
Packit 78deda
 * 2-25-98
Packit 78deda
 *
Packit 78deda
 * Mar 18, 1998  Jim Peterson  <jspeter@birch.ee.vt.edu>
Packit 78deda
 *
Packit 78deda
 *     Restructured to encapsulate more of the PBM handling.
Packit 78deda
 */
Packit 78deda
Packit 78deda
#include <stdio.h>
Packit 78deda
#include <stdlib.h>
Packit 78deda
#include <string.h>
Packit 78deda
Packit 78deda
#include "ppapbm.h"
Packit 78deda
Packit 78deda
int make_pbm_stat(pbm_stat* pbm,FILE* fptr)
Packit 78deda
{
Packit 78deda
  char line[1024];
Packit 78deda
Packit 78deda
  pbm->fptr=fptr;
Packit 78deda
  pbm->version=none;
Packit 78deda
  pbm->current_line=0;
Packit 78deda
  pbm->unread = 0;
Packit 78deda
Packit 78deda
  if (fgets (line, 1024, fptr) == NULL)
Packit 78deda
    return 0;
Packit 78deda
  line[strlen(line)-1] = 0;
Packit 78deda
Packit 78deda
  if(!strcmp(line,"P1")) pbm->version=P1;
Packit 78deda
  if(!strcmp(line,"P4")) pbm->version=P4;
Packit 78deda
  if(pbm->version == none)
Packit 78deda
  {
Packit 78deda
    fprintf(stderr,"pbm_readheader(): unknown PBM magic '%s'\n",line);
Packit 78deda
    return 0;
Packit 78deda
  }
Packit 78deda
Packit 78deda
  do
Packit 78deda
    if (fgets (line, 1024, fptr) == NULL)
Packit 78deda
      return 0;
Packit 78deda
  while (line[0] == '#');
Packit 78deda
Packit 78deda
  if (2 != sscanf (line, "%d %d", &pbm->width, &pbm->height))
Packit 78deda
    return 0;
Packit 78deda
Packit 78deda
  return 1;
Packit 78deda
}
Packit 78deda
Packit 78deda
static int getbytes(FILE *fptr,int width,unsigned char* data)
Packit 78deda
{
Packit 78deda
  unsigned char mask,acc,*place;
Packit 78deda
  int num;
Packit 78deda
Packit 78deda
  if(!width) return 0;
Packit 78deda
  for(mask=0x80, acc=0, num=0, place=data; num
Packit 78deda
  {
Packit 78deda
    switch(getc(fptr))
Packit 78deda
    {
Packit 78deda
    case EOF:      
Packit 78deda
      return 0;
Packit 78deda
    case '1':
Packit 78deda
      acc|=mask;
Packit 78deda
      /* fall through */
Packit 78deda
    case '0':
Packit 78deda
      mask>>=1;
Packit 78deda
      num++;
Packit 78deda
      if(!mask) /* if(num%8 == 0) */
Packit 78deda
      {
Packit 78deda
	*place++ = acc;
Packit 78deda
	acc=0;
Packit 78deda
	mask=0x80;
Packit 78deda
      }
Packit 78deda
    }
Packit 78deda
  }
Packit 78deda
  if(width%8)
Packit 78deda
    *place=acc;
Packit 78deda
  return 1;
Packit 78deda
}
Packit 78deda
Packit 78deda
/* Reads a single line into data which must be at least (pbm->width+7)/8
Packit 78deda
   bytes of storage */
Packit 78deda
int pbm_readline(pbm_stat* pbm,unsigned char* data)
Packit 78deda
{
Packit 78deda
  int tmp,tmp2;
Packit 78deda
Packit 78deda
  if(pbm->current_line >= pbm->height) return 0;
Packit 78deda
Packit 78deda
  if (pbm->unread)
Packit 78deda
    {
Packit 78deda
      memcpy (data, pbm->revdata, (pbm->width+7)/8);
Packit 78deda
      pbm->current_line++;
Packit 78deda
      pbm->unread = 0;
Packit 78deda
      free (pbm->revdata);
Packit 78deda
      pbm->revdata = NULL;
Packit 78deda
      return 1;
Packit 78deda
    }
Packit 78deda
Packit 78deda
  switch(pbm->version)
Packit 78deda
  {
Packit 78deda
  case P1:
Packit 78deda
    if(getbytes(pbm->fptr,pbm->width,data))
Packit 78deda
    {
Packit 78deda
      pbm->current_line++;
Packit 78deda
      return 1;
Packit 78deda
    }
Packit 78deda
    return 0;
Packit 78deda
Packit 78deda
  case P4:
Packit 78deda
    tmp=(pbm->width+7)/8;
Packit 78deda
    tmp2=fread(data,1,tmp,pbm->fptr);
Packit 78deda
    if(tmp2 == tmp)
Packit 78deda
    {
Packit 78deda
      pbm->current_line++;
Packit 78deda
      return 1;
Packit 78deda
    }
Packit 78deda
    fprintf(stderr,"pbm_readline(): error reading line data (%d)\n",tmp2);
Packit 78deda
    return 0;
Packit 78deda
Packit 78deda
  default:
Packit 78deda
    fprintf(stderr,"pbm_readline(): unknown PBM version\n");
Packit 78deda
    return 0;
Packit 78deda
  }
Packit 78deda
}
Packit 78deda
Packit 78deda
/* push a line back into the buffer; we read too much! */
Packit 78deda
void pbm_unreadline (pbm_stat *pbm, void *data)
Packit 78deda
{
Packit 78deda
  /* can only store one line in the unread buffer */
Packit 78deda
  if (pbm->unread)
Packit 78deda
    return;
Packit 78deda
Packit 78deda
  pbm->unread = 1;
Packit 78deda
  pbm->revdata = malloc ((pbm->width+7)/8);
Packit 78deda
  memcpy (pbm->revdata, data, (pbm->width+7)/8);
Packit 78deda
  pbm->current_line--;
Packit 78deda
}