|
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 |
}
|