/* -*-C-*-

$Id: ptrvec.c,v 1.3 1999/01/03 05:33:56 cph Exp $

Copyright (C) 1990-1999 Massachusetts Institute of Technology

This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2 of the License, or (at
your option) any later version.

This program is distributed in the hope that it will be useful, but
WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
General Public License for more details.

You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
*/

#include "outf.h"
#include "dstack.h"

static PTR
DEFUN (xmalloc, (length), unsigned int length)
{
  extern PTR EXFUN (malloc, (unsigned int length));
  PTR result = (malloc (length));
  if (result == 0)
    {
      outf_fatal ("malloc: memory allocation failed\n");
      outf_flush_fatal ();
      abort ();
    }
  return (result);
}

static PTR
DEFUN (xrealloc, (ptr, length), PTR ptr AND unsigned int length)
{
  extern PTR EXFUN (realloc, (PTR ptr, unsigned int length));
  PTR result = (realloc (ptr, length));
  if (result == 0)
    {
      outf_fatal ("realloc: memory allocation failed\n");
      outf_flush_fatal ();
      abort ();
    }
  return (result);
}

Tptrvec
DEFUN (ptrvec_allocate, (length), Tptrvec_length length)
{
  Tptrvec ptrvec = (xmalloc (sizeof (struct struct_ptrvec)));
  (ptrvec -> length) = length;
  (ptrvec -> elements) =
    ((length > 0) ? (xmalloc (length * (sizeof (PTR)))) : 0);
  return (ptrvec);
}

void
DEFUN (ptrvec_deallocate, (ptrvec), Tptrvec ptrvec)
{
  if ((ptrvec -> length) > 0)
    free (ptrvec -> elements);
  free (ptrvec);
}

void
DEFUN (ptrvec_set_length, (ptrvec, length),
       Tptrvec ptrvec AND
       Tptrvec_length length)
{
  (ptrvec -> length) = length;
  (ptrvec -> elements) =
    ((length > 0)
     ? (xrealloc ((ptrvec -> elements), (length * (sizeof (PTR)))))
     : 0);
}

Tptrvec
DEFUN (ptrvec_copy, (ptrvec), Tptrvec ptrvec)
{
  Tptrvec_length length = (PTRVEC_LENGTH (ptrvec));
  Tptrvec result = (ptrvec_allocate (length));
  PTR * scan_source = (PTRVEC_START (ptrvec));
  PTR * end_source = (scan_source + length);
  PTR * scan_result = (PTRVEC_START (result));
  while (scan_source < end_source)
    (*scan_result++) = (*scan_source++);
  return (result);
}

void
DEFUN (ptrvec_adjoin, (ptrvec, element), Tptrvec ptrvec AND PTR element)
{
  Tptrvec_length length = (PTRVEC_LENGTH (ptrvec));
  ptrvec_set_length (ptrvec, (length + 1));
  (PTRVEC_REF (ptrvec, length)) = element;
}

int
DEFUN (ptrvec_memq, (ptrvec, element), Tptrvec ptrvec AND PTR element)
{
  PTR * scan = (PTRVEC_START (ptrvec));
  PTR * end = (scan + (PTRVEC_LENGTH (ptrvec)));
  while (scan < end)
    if (element == (*scan++))
      return (1);
  return (0);
}

void
DEFUN (ptrvec_move_left,
       (source, source_start, source_end, target, target_start),
       Tptrvec source AND
       Tptrvec_index source_start AND
       Tptrvec_index source_end AND
       Tptrvec target AND
       Tptrvec_index target_start)
{
  PTR * scan_source = (PTRVEC_LOC (source, source_start));
  PTR * end_source = (PTRVEC_LOC (source, source_end));
  PTR * scan_target = (PTRVEC_LOC (target, target_start));
  while (scan_source < end_source)
    (*scan_target++) = (*scan_source++);
}

void
DEFUN (ptrvec_move_right,
       (source, source_start, source_end, target, target_start),
       Tptrvec source AND
       Tptrvec_index source_start AND
       Tptrvec_index source_end AND
       Tptrvec target AND
       Tptrvec_index target_start)
{
  PTR * end_source = (PTRVEC_LOC (source, source_start));
  PTR * scan_source = (PTRVEC_LOC (source, source_end));
  PTR * scan_target =
    (PTRVEC_LOC (target, (target_start + (source_end - source_start))));
  while (scan_source > end_source)
    (*--scan_target) = (*--scan_source);
}


syntax highlighted by Code2HTML, v. 0.9.1