view oldclasslib/include/stack.h @ 15:f5acaf0c8a29

Don't cast through "volatile int". Causes a gcc warning nowadays. XXX: should put something else back here to frighten the optimizer
author David A. Holland
date Tue, 31 May 2022 01:00:55 -0400
parents 13d2b8934445
children
line wrap: on
line source

/*
 * AnaGram, a System for Syntax Directed Programming
 *
 * Stack Class Definition
 *
 * Please note that the entire class is defined in this module. There
 * is no supplementary code module.
 *
 * Copyright 1993 Parsifal Software. All Rights Reserved.
 *
 * This software is provided 'as-is', without any express or implied
 * warranty.  In no event will the authors be held liable for any damages
 * arising from the use of this software.
 *
 * Permission is granted to anyone to use this software for any purpose,
 * including commercial applications, and to alter it and redistribute it
 * freely, subject to the following restrictions:
 *
 * 1. The origin of this software must not be misrepresented; you must not
 *    claim that you wrote the original software. If you use this software
 *    in a product, an acknowledgment in the product documentation would be
 *    appreciated but is not required.
 * 2. Altered source versions must be plainly marked as such, and must not be
 *    misrepresented as being the original software.
 * 3. This notice may not be removed or altered from any source distribution.
 */

#ifndef STACK_H
#define STACK_H

#include <assert.h>

// Some forward decls required in current C++
template <class T> class stack;
template <class T> stack <T> &reset(stack <T> &s);
template <class T> unsigned size(const stack <T> &s);


template <class T>
class stack {
protected:
  T *cs;                               // data storage
  unsigned *ls;                        // index (level) storage
  unsigned sx;                         // stack index
  unsigned lx;                         // level index
  unsigned sxmax;                      // max stack index
  unsigned lxmax;                      // max level index
  int copy_flag;                       // is it live or memorex?
public:


// Constructor

  stack(unsigned n, unsigned m) {      // (storage request, levels)
    cs = new T[n];                     // allocate data storage
    ls = new unsigned[m];              // allocate index storage
    lx = lxmax = m;                    // set up for pre-decrement push
    sx = sxmax = n;                    // post-increment pop
    ls[--lx] = sx;                     // init first level
    copy_flag = 0;                     // live
  }

  stack(unsigned n) {                  // (storage request)
    unsigned m = 1;                    // single level
    cs = new T[n];                     // allocate data storage
    ls = new unsigned[m];              // allocate index storage
    lx = lxmax = m;                    // set up for pre-decrement push
    sx = sxmax = n;                    // post-increment pop
    ls[--lx] = sx;                     // init first level
    copy_flag = 0;                     // live
  }

  stack(stack<T> &s) {
    *this = s;
    copy_flag++;                       // memorex
  }


// Destructor

  ~stack() {
    if (copy_flag) return;
    delete [] cs;
    delete [] ls;
  }


// Reset function
#ifdef __IBMCPP__
  friend stack<T> &reset(stack<T> &s);
#else
  friend stack<T> &reset<>(stack<T> &s);
#endif


// Change levels

  stack<T> &operator ++() {            // next level -- preincrement
    assert(copy_flag == 0);
    assert(lx);                        // not going to overflow
    ls[--lx] = sx;                     // push storage index
    return *this;
  }

  stack<T> operator ++(int);           // next level -- postincrement

  stack<T> &operator --() {            // previous level -- predecrement
    assert(copy_flag == 0);
    sx = ls[lx++];                     // restore storage index
    assert(lx < lxmax);                // no underflow
    return *this;
  }

  stack<T> operator --(int);         // previous level -- postdecrement


// Push data

  stack<T> &operator << (T t) {        // push data
    assert(copy_flag == 0);
    assert(sx && lx < lxmax);          // everything's ok
    cs[--sx] = t;                      // save it
    return *this;
  }


// Pop data

  stack<T> &operator >> (T &t) {       // pop data
    assert(copy_flag == 0);
    assert(lx < lxmax && sx < ls[lx]); // all's well
    t = cs[sx++];                      // get it
    return *this;
  }


// Access stack element

  T &operator[] (unsigned k) {         // access stack element
    assert(lx < lxmax && sx + k < ls[lx]);  // legitimate index
    return cs[sx+k];                   // reference data
  }


// Pointer to top element

// See README.txt
#if 0
  operator T * () {
    return &cs[sx];
  }
#endif
  T *top() {
    return &cs[sx];
  }


// Get size of stack

#ifdef __IBMCPP__
  friend unsigned size(const stack <T> &s);        // size at current level
#else
  friend unsigned size<>(const stack <T> &s);      // size at current level
#endif

};

template <class T>
inline stack <T> &reset(stack <T> &s) {
  assert(s.copy_flag == 0);
  s.lx = s.lxmax;                    // set up for pre-decrement push
  s.sx = s.sxmax;                    // post-increment pop
  s.ls[--s.lx] = s.sx;               // init first level
  return s;
}

template <class T>
inline unsigned size(const stack <T> &s) {     // size at current level
  return s.ls[s.lx] - s.sx;
}


template <class T>
stack<T> stack<T>::operator ++(int) {   // next level -- postintcrement
  stack<T> cpy(*this);                  // make copy for return value
  assert(copy_flag == 0);
  assert(lx);
  ls[--lx] = sx;
  return cpy;
}

template <class T>
stack<T> stack<T>::operator --(int) {    // previous level -- postdecrement
  stack <T> cpy(*this);                  // make copy for return value
  assert(copy_flag == 0);
  sx = ls[lx++];                     // restore storage index
  assert(lx < lxmax);                // no underflow
  return cpy;
}
#endif