Sophie

Sophie

distrib > Mandriva > 2009.1 > x86_64 > by-pkgid > bbb305e6fb2f4e8ebc72959c802137bf > files > 3

dcraw-8.93-1mdv2009.1.src.rpm

/*
   Because they are parsed from the end, Canon CRW files
   become unreadable if garbage data is appended to them, as
   often happens when files are recovered from damaged media.
   This program truncates CRW files to the correct size.

   Copyright 2005 by Dave Coffin, dcoffin a cybercom o net
   Free for all uses.

   $Revision$
   $Date$
 */

#include <stdio.h>
#include <stdlib.h>
#include <string.h>

unsigned char *buffer;

int get4 (int i)
{
  if (buffer[0] == 'I')
    return buffer[i+3] << 24 | buffer[i+2] << 16 | buffer[i+1] << 8 | buffer[i];
  else
    return buffer[i] << 24 | buffer[i+1] << 16 | buffer[i+2] << 8 | buffer[i+3];
}

int main (int argc, char **argv)
{
  int arg, size, end, diff, status=1;
  unsigned char *fname;
  FILE *fp;

  if (argc == 1)
    fprintf (stderr, "Usage:  %s crw_0001.crw crw_0002.crw ...\n", argv[0]);

  for (arg=1; arg < argc; arg++) {
    status = 1;
    fp = fopen (argv[arg], "rb");
    fseek (fp, 0, SEEK_END);
    size = ftell(fp);
    buffer = malloc (size + strlen(argv[arg]) + 10);
    if (!buffer) {
      fprintf (stderr, "Cannot allocate memory!\n");
      return 2;
    }
    fname = buffer + size;
    sprintf (fname, "%s.clean", argv[arg]);
    fseek (fp, 0, SEEK_SET);
    fread (buffer, 1, size, fp);
    fclose (fp);
    if (strncmp (buffer, "II\x1a\0\0\0HEAPCCDR", 14) &&
	strncmp (buffer, "MM\0\0\0\x1aHEAPCCDR", 14)) {
      fprintf (stderr, "%s is not a CRW file!\n", argv[arg]);
      free (buffer);
      continue;
    }
    for (end=size; end > 0xa0000; end--) {
      diff = end - get4(end-4);
      if (diff > 50 && diff < 120 && diff % 10 == 2) {
	status = 0;
	break;
      }
    }
    if (status)
      fprintf (stderr, "Failed to clean %s\n", argv[arg]);
    else {
      if ((fp = fopen (fname, "wb"))) {
	fprintf (stderr, "Writing %s\n", fname);
	fwrite (buffer, 1, end, fp);
	fclose (fp);
      } else {
	perror (fname);
	status = 1;
      }
    }
    free (buffer);
  }
  return status;
}