Anforderungen  |   Konzepte  |   Entwurf  |   Entwicklung  |   Qualitätssicherung  |   Lebenszyklus  |   Steuerung
 
 
 
 


Quelle  diffredcos.c   Sprache: C

 
/* diffredcos.c  2/8/95
 * 9/1/98 type of generator changed from char to `gen'
 * version of diff_reduce in cosets context.
 * Copied and edited from diffreduce.c
 *
 * Copied from old automata package and edited.
 * This file contains the procedure for reducing a word using a
 * word-difference machine *wd_fsa (defined externally).
 */

#include "defs.h"
#include "fsa.h"
#include "rws.h"
#include "externals.h"

#define MAXV 65536 /* The maximum number of vertices allowed. */

/* Functions defined in this file: */

/* w is the word to be reduced using the word-difference machine  *wd_fsa.
 * It is assumed that wd_fsa->table->table_data_dptr is set up.
 * This function allocates its own space.
 * NOTE: No checks on the validity of the word are carried out.
 * For the cosets version, the temporary word prefix is used to keep
 * the prefix in the subgroup.
 * w must either be a word in the generators of G, in which case it is simply
 * reduced as this, just as it would be in word_reduce,
 * or it must include the separator symbol and be followed by word
 * in the generators of G, when it will be reduced as a coset.
 * It must have enough space allocated to hold the reduced word.
 */

int diff_reduce_cos(gen *w, reduction_struct *rs_wd)
{
  int ndiff, ngens, identity, padsymbol, wordlen, ***difftab, gct, *gpref,
      level, gen1, gen2, diff, diffct = 0, newdiff, olen, nlen, i, j = 0;
  boolean deqi, donesub, *cf, coset = FALSE;
  gen *ptr, *ptre, *ptrc;
  gen prefix[8192];
  int maxv = MAXV;
  fsa *wd_fsa = rs_wd->wd_fsa;
  int separator = rs_wd->separator;
  struct vertexd {
    gen genno;
    int diffno;
    int sublen;
    struct vertexd *backptr;
  } * gptr, *ngptr, *substruc;

  /* vertexd is the structure used to store a vertex in the graph of strings
     for possible substitution. The components are as follows.
     backptr - points back to another vertexd, or to zero.
     genno  - the number of the generator at the end of the string.
     diffno - the word difference number of the string defined by following
              backptr back to zero (using genno), relative to the corresponding
              part of the word being reduced.
     sublen - plus or minus the length of this string. sublen is positive if and
              only if the string lexicographically precedes the
              corresponding part of the word being reduced.
     sublen is put in to save time.
     NOTE - in the cosets version, sublen is 0 if the substitution starts at
     the beginning of the word, there is a prefix, and both strings in the
     G-gens are equal up to that point. Another essential component of a vertexd
     is its level (i.e. the length of the string got by chasing back to the
     beginning of the word) but we always calculate this, using the integers
     defined by gpref. (See below))
  */


  if (wd_fsa->alphabet->type != PRODUCT || wd_fsa->alphabet->arity != 2) {
    fprintf(stderr, "Error: diff_reduce_cos must be called with a "
                    "word-difference machine.\n");
    return -1;
  }
  /* First we see if the word contains the separator, and move w to it
   * if it does.
   */

  ptr = w - 1;
  while (*(++ptr))
    if (*ptr == separator) {
      w = ptr;
      break;
    }
  ndiff = wd_fsa->states->size;
  ngens = wd_fsa->alphabet->base->size;
  identity = wd_fsa->initial[1];
  padsymbol = ngens + 1;
  wordlen = genstrlen(w);
  if (wordlen == 0)
    return 0;
  difftab = wd_fsa->table->table_data_dptr;
  w = w - 1; /* since code below assumes word is from w[1] .. w[wordlen]. */

  tmalloc(cf, boolean, ndiff + 1);
  /* cf is used as a characteristic function, when constructing a subset of the
    set  D  of word differences.
  */


  tmalloc(gpref, int, wordlen + 1);
  gct = -1;
  gpref[0] = -1;
  /* gpref[n]+1 is the number of vertices that have been defined after reading
    the first n elements of the word. These vertices are
    gptr[0],...,gptr[gpref[n]]. We start by allocating space for maxv vertices.
  */


  tmalloc(gptr, struct vertexd, maxv);
  /* First check that the separator only occurs once. */
  level = 1;
  while (++level <= wordlen)
    if (w[level] == separator) {
      fprintf(
          stderr,
          "Error: in diff_reduce_cos, subgroup symbol can only occur once.\n");
      return -1;
    }
  if (w[1] == separator) {
    *prefix = 0;
    coset = TRUE;
  }

  /* Now we start reading the word. */
  level = 0;
  while (++level <= wordlen) {
    /* Read the element of the word at position level. */
    gen1 = w[level];

    if (gen1 == separator) {
      /* We put in the start states on the list for this level. */
      for (i = 2; i <= wd_fsa->num_initial; i++) {
        gct++;
        substruc = gptr + gct;
        substruc->genno = 0;
        substruc->diffno = wd_fsa->initial[i];
        substruc->sublen = 0;
        substruc->backptr = 0;
      }
      gpref[1] = gct;
      continue;
    }

    for (i = 1; i <= ndiff; i++)
      cf[i] = FALSE;

    /* The next loop is over the identity and the subset of D defined at the
       previous level, level-1.
    */

    diff = identity;
    while (1) {
      deqi = diff == identity;
      /* First look for a possible substitution of a shorter string */
      newdiff = dense_dtarget(difftab, gen1, padsymbol, diff);
      if (newdiff == identity) {
        /* Make substitution and reduce length of word by 1. */
        i = level - 1;
        if (!deqi) {
          substruc = gptr + diffct;
          if (!coset || i > 1) {
            do {
              w[i] = substruc->genno;
              substruc = substruc->backptr;
              i--;
            } while (substruc && (!coset || i > 1));
          }
          if (coset && substruc && i == 1) { /* we have a prefix */
            genstrcat(prefix, wd_fsa->states->words[substruc->diffno]);
            diff_reduce_cos(prefix, rs_wd);
          }
        }
        for (j = level; j < wordlen; j++)
          w[j] = w[j + 1];
        w[wordlen] = 0;
        wordlen--;

        /* Whenever we make a substitution, we have to go back one level more
           than expected, because of our policy of looking ahead for
           substitutions that reduce the length by 2.
        */

        level = i > 0 ? i - 1 : i;
        gct = gpref[level];
        break;
      }
      else if (newdiff && level < wordlen) {
        j = dense_dtarget(difftab, w[level + 1], padsymbol, newdiff);
        if (j == identity)
        /* Make substitution and reduce length of word by 2. */
        {
          i = level - 1;
          if (!deqi) {
            substruc = gptr + diffct;
            if (!coset || i > 1) {
              do {
                w[i] = substruc->genno;
                substruc = substruc->backptr;
                i--;
              } while (substruc && (!coset || i > 1));
            }
            if (coset && substruc && i == 1) { /* we have a prefix */
              genstrcat(prefix, wd_fsa->states->words[substruc->diffno]);
              diff_reduce_cos(prefix, rs_wd);
            }
          }
          for (j = level; j < wordlen - 1; j++)
            w[j] = w[j + 2];
          w[wordlen - 1] = 0;
          wordlen -= 2;
          level = i > 0 ? i - 1 : i;
          gct = gpref[level];
          break;
        }
      }

      donesub = FALSE;
      /* Now we loop over the generator that is a candidate for substitution
         at this point.
      */

      for (gen2 = 1; gen2 <= ngens; gen2++) {
        newdiff = dense_dtarget(difftab, gen1, gen2, diff);
        if (newdiff) {
          if (newdiff == identity) {
            if (deqi || level == 1) {
              if (gen2 < gen1) {
                w[level] = gen2;
                if (level == 1) { /* we have a prefix */
                  genstrcat(prefix, wd_fsa->states->words[diff]);
                  diff_reduce_cos(prefix, rs_wd);
                }
                level = level > 1 ? level - 2 : level - 1;
                gct = gpref[level];
                donesub = TRUE;
                break;
              }
            }
            else if (gptr[diffct].sublen > 0 ||
                     (gptr[diffct].sublen == 0 && gen2 < gen1)) {
              /* Make a substitution (by a string of equal length). */
              w[level] = gen2;
              i = level - 1;
              substruc = gptr + diffct;
              if (!coset || i > 1) {
                do {
                  w[i] = substruc->genno;
                  substruc = substruc->backptr;
                  i--;
                } while (substruc && (!coset || i > 1));
              }
              if (coset && substruc && i == 1) { /* we have a prefix */
                genstrcat(prefix, wd_fsa->states->words[substruc->diffno]);
                diff_reduce_cos(prefix, rs_wd);
              }
              level = i > 0 ? i - 1 : i;
              gct = gpref[level];
              donesub = TRUE;
              break;
            }
          }
          else {
            if (cf[newdiff])
              /* We have this word difference stored already, but we will check
                 to see if the current string precedes the existing one.
              */

              for (i = gpref[level - 1] + 1;; i++) {
                substruc = gptr + i;
                if (substruc->diffno == newdiff) {
                  olen = substruc->sublen;
                  if (!deqi)
                    j = gptr[diffct].sublen;
                  if (deqi || j == 0) {
                    nlen = gen2 < gen1 ? 1 : gen2 == gen1 ? 0 : -1;
                  }
                  else {
                    nlen = j > 0 ? j + 1 : j - 1;
                  }
                  if (nlen > olen) {
                    /* The new string is better than the existing one */
                    substruc->genno = gen2;
                    substruc->sublen = nlen;
                    substruc->backptr = deqi ? 0 : gptr + diffct;
                  }
                  break;
                }
              }
            else
            /* This is a new word difference at this level, so we define a new
               vertexd in graph.
            */

            {
              gct++;
              if (gct >= maxv) {
                /* We need more space for vertices. Allocate twice the preceding
                   space and copy existing data.
                */

                tmalloc(ngptr, struct vertexd, 2 * maxv);
                if (kbm_print_level >= 3)
                  printf(" #Allocating more space in diff_reduce_cos.\n");
                for (i = 0; i < maxv; i++) {
                  ngptr[i].genno = gptr[i].genno;
                  ngptr[i].diffno = gptr[i].diffno;
                  ngptr[i].sublen = gptr[i].sublen;
                  substruc = gptr[i].backptr;
                  if (substruc == 0)
                    ngptr[i].backptr = 0;
                  else
                    for (j = i - 1;; j--)
                      if (substruc == gptr + j) {
                        ngptr[i].backptr = ngptr + j;
                        break;
                      }
                }
                tfree(gptr);
                gptr = ngptr;
                maxv *= 2;
              }
              /* Define the new vertexd. */
              substruc = gptr + gct;
              if (!deqi)
                j = gptr[diffct].sublen;
              if (deqi || j == 0) {
                nlen = gen2 < gen1 ? 1 : gen2 == gen1 ? 0 : -1;
              }
              else {
                nlen = j > 0 ? j + 1 : j - 1;
              }
              substruc->genno = gen2;
              substruc->diffno = newdiff;
              substruc->sublen = nlen;
              substruc->backptr = deqi ? 0 : gptr + diffct;
              cf[newdiff] = TRUE;
            }
          }
        }
      } /*End of loop over gen2 */

      if (donesub)
        break;

      /* Go on to next word difference from previous level. */
      if (diff == identity) {
        diffct = level == 1 ? 0 : gpref[level - 2] + 1;
      }
      else
        diffct++;
      if (diffct > gpref[level - 1])
        break;
      diff = gptr[diffct].diffno;
    } /* end of loop over word differences at previous level */

    gpref[level] = gct;
  }

  /* Finally, in the coset case, rewrite the full-word as prefix*separator*word
   */

  w = w + 1; /* put it back in the right place ! */
  if (coset && genstrlen(prefix) > 0) {
    ptr = w;
    ptre = w + genstrlen(w);
    ptrc = ptre + genstrlen(prefix);
    while (ptre >= ptr)
      *(ptrc--) = *(ptre--);
    ptr = prefix;
    ptre = ptr + genstrlen(prefix) - 1;
    while (ptre >= ptr)
      *(ptrc--) = *(ptre--);
  }

  tfree(gptr);
  tfree(cf);
  tfree(gpref);
  return 0;
}

Messung V0.5
C=94 H=89 G=91

¤ Dauer der Verarbeitung: 0.6 Sekunden  ¤

*© Formatika GbR, Deutschland






Wurzel

Suchen

Beweissystem der NASA

Beweissystem Isabelle

NIST Cobol Testsuite

Cephes Mathematical Library

Wiener Entwicklungsmethode

Haftungshinweis

Die Informationen auf dieser Webseite wurden nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit, noch Qualität der bereit gestellten Informationen zugesichert.

Bemerkung:

Die farbliche Syntaxdarstellung und die Messung sind noch experimentell.






                                                                                                                                                                                                                                                                                                                                                                                                     


Neuigkeiten

     Aktuelles
     Motto des Tages

Software

     Produkte
     Quellcodebibliothek

Aktivitäten

     Artikel über Sicherheit
     Anleitung zur Aktivierung von SSL

Muße

     Gedichte
     Musik
     Bilder

Jenseits des Üblichen ....
    

Besucherstatistik

Besucherstatistik

Monitoring

Montastic status badge