view anagram/agcore/checksum.cpp @ 20:bb115deb6fb2

Improve agfiles rule. (1) It didn't depend on $(AGCL) and it absolutely should have. (2) allow AGFORCE=1 to make it rebuild whether or not it looks out of date. (3) Document this.
author David A. Holland
date Mon, 13 Jun 2022 00:02:15 -0400
parents 13d2b8934445
children
line wrap: on
line source

/*
 * AnaGram, A System for Syntax Directed Programming
 * Copyright 1993-1999 Parsifal Software. All Rights Reserved.
 * Copyright 2006 David A. Holland. All Rights Reserved.
 * See the file COPYING for license and usage terms.
 *
 * checksum.cpp - self-checksum module
 */

#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <limits.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include "port.h"

#ifdef AG_ON_WINDOWS
#include <windows.h>
#endif

#include "agstring.h"
#include "assert.h"
#include "checksum.h"
#include "file.h"

//#define INCLUDE_LOGGING
#include "log.h"


/*
 * Fletcher's check-sum routines for AnaGram
 *
 * Freely adapted from routines published in Dr. Dobbs Journal
 * May 1992, p. 64
 */

#define BLOCK_SIZE 0X4000U

static AgString basedir;

static void checkSum(int fh, int offset, u_long *res_sum, u_long *res_len) {
  LOGSECTION("checkSum");
  char *buf;
  unsigned short k1, k2;

  *res_sum = 0;
  *res_len = 0;

  //LOGV(fh);

  if (fh < 0) {
    return;
  }
  lseek(fh, 0, 2);
  int length = tell(fh);
  *res_len = length;
  //LOGV(length);
  //buf = ALLOCATE(BLOCK_SIZE + 1, char);
  buf = new char[BLOCK_SIZE + 1];

  k1 = (unsigned short) (length >> 16);
  k2 = (unsigned short) length;

  if (offset) {
    lseek(fh, offset+512, 0);
    length -= offset+512;
    //LOGV(length);
  }
  else {
    lseek(fh,0,0);
  }

  if (k1 == 0) k1++;
  if (k2 == 0) k2++;

  while (length > 0) {
    unsigned n;
    unsigned ni;
    unsigned i;
    unsigned short *b;

    n = BLOCK_SIZE;
    if ((long) n > length) {
      n = (unsigned short) length;
    }
    n = read(fh, buf, n);
    if (n == 0) {
      break;
    }
    ni = (n+1)/2;
    i = 0;
    b = (unsigned short *) buf;
    buf[n] = 0;
    while (i< ni) {
      k1 += b[i];
      if (k1 < b[i]) {
	k1++;
      }
      k2 += k1;
      if (k2 < k1) {
	k2++;
      }
      i++;
    }
    length -= n;
    if (length <= 0 && offset!=0) {
      //LOGV(k1) LCV(k2);
      length = offset;
      offset = 0;
      lseek(fh, 0, 0);
    }
  }
  k1 ^= (unsigned short) -1;
  k2 ^= (unsigned short) -1;
  *res_sum = 65535L*k2 + k1;

  LOGV(*res_sum);
  //DEALLOCATE(buf);
  delete [] buf;
}

#ifdef AG_ON_WINDOWS
static int open_binary(summable what) {
  char buf[_MAX_PATH];
  HMODULE module;

  switch (what) {
    case SUM_AG1:  module = GetModuleHandle("ag1"); break;
    case SUM_AG:   module = 0; break;
    case SUM_AGCL: module = 0; break;
  }

  GetModuleFileName(module, buf, sizeof(buf)-2);

  if (what != SUM_AG1) {
    size_t len = strlen(buf);
    if (len >= 6 && !stricmp(buf+len-6, "ag.exe") && what == SUM_AGCL) {
      strcpy(buf+len-6, "agcl.exe");
    }
    else if (len >= 8 && !stricmp(buf+len-8, "agcl.exe") && what == SUM_AG) {
      strcpy(buf+len-8, "ag.exe");
    }
  }

  return open_shared_denywrite(buf, O_BINARY|O_RDONLY);
}
#endif

#ifdef AG_ON_UNIX
static int open_binary(summable what) {
  char buf[PATH_MAX];
  const char *name = NULL;
  switch (what) {
    case SUM_AG1:  name = "ag1.so"; break;
    case SUM_AG:   name = "ag"; break;
    case SUM_AGCL: name = "agcl"; break;
  }

  snprintf(buf, sizeof(buf), "%s/%s", basedir.pointer(), name);
  return open(buf, O_RDONLY);
}
#endif

static int open_summable(summable what) {
  switch (what) {
    case SUM_AG1:
    case SUM_AG:
    case SUM_AGCL:
      return open_binary(what);
  }
  return -1;
}

static void close_summable(summable what, int fd) {
  switch (what) {
    case SUM_AG1:
    case SUM_AG:
    case SUM_AGCL:
      close(fd);
      break;
  }
}

void observeSum(sumentry *s) {
  LOGSECTION("observeSum");

  int fd = open_summable(s->what);
  if (fd >= 0) {
    LOGV(fd);
    checkSum(fd, s->offset, &s->observed.sum, &s->observed.length);
    close_summable(s->what, fd);
  }
  else {
    s->observed.sum = 0;
    s->observed.length = 0;
  }
}

void sum_remember_argv0(const char *argv0) {
  LOGSECTION("sum_remember_argv0");
  LOGV(argv0);
  AgString av0(argv0);

#ifdef AG_ON_WINDOWS
  basedir = av0.lastCut("\\/:").leftX();
  LOGV(basedir);
#endif

#ifdef AG_ON_UNIX
  /*
   * Not as simple on Unix - we get whatever the parent process sends
   * us, which is typically what the user typed into the shell to
   * invoke the program. That is, it might be an absolute path, a
   * relative path, or no path at all, and in the latter case we need
   * to search $PATH. Blah.
   *
   * Note that since both argv[0] and $PATH are under the user's
   * control, if the user is adversarial we can't count on finding the
   * same file that we're actually executing from. Fortunately we
   * don't care about that now AG is open source.
   */

  if (argv0[0]=='/') {
    /* absolute path */
    basedir = av0.lastCut("\\/:").leftX();
    LOGV(basedir);
  }
  else if (strrchr(argv0, '/')!=NULL) {
    /* relative path - good enough; we don't chdir before summing is done */
    basedir = av0.lastCut("\\/:").leftX();
    LOGV(basedir);
  }
  else {
    /* nothing */
    struct stat sb;
    const char *p = getenv("PATH");
    LOGV(p);
    if (p) {
      AgString path(p); // copy it
      for (char *s = strtok(path.pointer(), ":"); s; s = strtok(NULL, ":")) {
	char tmp[PATH_MAX];
	snprintf(tmp, sizeof(tmp), "%s/%s", s, argv0);
	LOGV(tmp);
	if (stat(tmp, &sb)==0) {
	  basedir = s;	// copy it
	  LOGV(basedir);
	  return;
	}
      }
    }

    /* oh well - let's make a guess */
    basedir = "/usr/local/lib/anagram";
    LOGV(basedir);
  }
#endif
}