vecSet.h 10.7 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11
/**CFile****************************************************************

  FileName    [vecSet.h]

  SystemName  [ABC: Logic synthesis and verification system.]

  PackageName [SAT solvers.]

  Synopsis    [Multi-page dynamic array.]

  Author      [Alan Mishchenko]
12

13 14 15 16 17 18 19
  Affiliation [UC Berkeley]

  Date        [Ver. 1.0. Started - June 20, 2005.]

  Revision    [$Id: vecSet.h,v 1.00 2005/06/20 00:00:00 alanmi Exp $]

***********************************************************************/
20

21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42
#ifndef ABC__sat__bsat__vecSet_h
#define ABC__sat__bsat__vecSet_h


////////////////////////////////////////////////////////////////////////
///                          INCLUDES                                ///
////////////////////////////////////////////////////////////////////////

#include <stdio.h>

ABC_NAMESPACE_HEADER_START


////////////////////////////////////////////////////////////////////////
///                         PARAMETERS                               ///
////////////////////////////////////////////////////////////////////////

////////////////////////////////////////////////////////////////////////
///                         BASIC TYPES                              ///
////////////////////////////////////////////////////////////////////////

// data-structure for logging entries
43 44 45 46 47
// memory is allocated in 2^nPageSize word-sized pages
// the first two 'words' of each page are used for bookkeeping
// the first 'word' of bookkeeping data stores the word limit
// the second 'word' of bookkeeping data stores the shadow word limit
// (the shadow word limit is only used during garbage collection)
48 49

typedef struct Vec_Set_t_ Vec_Set_t;
50
struct Vec_Set_t_
51
{
52 53 54 55 56 57 58
    int               nPageSize;    // page size
    unsigned          uPageMask;    // page mask
    int               nEntries;     // entry count
    int               iPage;        // current page
    int               iPageS;       // shadow page
    int               nPagesAlloc;  // page count allocated
    word **           pPages;       // page pointers
59
}; 
60 61 62 63 64

////////////////////////////////////////////////////////////////////////
///                      MACRO DEFINITIONS                           ///
////////////////////////////////////////////////////////////////////////

65 66
static inline int     Vec_SetHandPage( Vec_Set_t * p, int h )    { return h >> p->nPageSize;      }
static inline int     Vec_SetHandShift( Vec_Set_t * p, int h )   { return h & p->uPageMask;       }
67
static inline int     Vec_SetWordNum( int nSize )                { return (nSize + 1) >> 1;       }
68

69 70
//static inline word *  Vec_SetEntry( Vec_Set_t * p, int h )       { assert(Vec_SetHandPage(p, h) >= 0 && Vec_SetHandPage(p, h) <= p->iPage); assert(Vec_SetHandShift(p, h) >= 2 && Vec_SetHandShift(p, h) < (1 << p->nPageSize)); return p->pPages[Vec_SetHandPage(p, h)] + Vec_SetHandShift(p, h);     }
static inline word *  Vec_SetEntry( Vec_Set_t * p, int h )       { return p->pPages[Vec_SetHandPage(p, h)] + Vec_SetHandShift(p, h);                     }
71 72 73
static inline int     Vec_SetEntryNum( Vec_Set_t * p )           { return p->nEntries;            }
static inline void    Vec_SetWriteEntryNum( Vec_Set_t * p, int i){ p->nEntries = i;               }

74 75
static inline int     Vec_SetLimit( word * p )                   { return p[0];                   }
static inline int     Vec_SetLimitS( word * p )                  { return p[1];                   }
76

77 78
static inline int     Vec_SetIncLimit( word * p, int nWords )    { return p[0] += nWords;         }
static inline int     Vec_SetIncLimitS( word * p, int nWords )   { return p[1] += nWords;         }
79

80 81
static inline void    Vec_SetWriteLimit( word * p, int nWords )  { p[0] = nWords;                 }
static inline void    Vec_SetWriteLimitS( word * p, int nWords ) { p[1] = nWords;                 }
82

83 84
static inline int     Vec_SetHandCurrent( Vec_Set_t * p )        { return (p->iPage << p->nPageSize)  + Vec_SetLimit(p->pPages[p->iPage]);               }
static inline int     Vec_SetHandCurrentS( Vec_Set_t * p )       { return (p->iPageS << p->nPageSize) + Vec_SetLimitS(p->pPages[p->iPageS]);             }
85

86 87 88 89
static inline int     Vec_SetHandMemory( Vec_Set_t * p, int h )  { return Vec_SetHandPage(p, h) * (1 << (p->nPageSize+3)) + Vec_SetHandShift(p, h) * 8;  }
static inline int     Vec_SetMemory( Vec_Set_t * p )             { return Vec_SetHandMemory(p, Vec_SetHandCurrent(p));                                   }
static inline int     Vec_SetMemoryS( Vec_Set_t * p )            { return Vec_SetHandMemory(p, Vec_SetHandCurrentS(p));                                  }
static inline int     Vec_SetMemoryAll( Vec_Set_t * p )          { return (p->iPage+1) * (1 << (p->nPageSize+3));                                        }
90 91 92 93 94 95 96 97

// Type is the Set type
// pVec is vector of set
// nSize should be given by the user
// pSet is the pointer to the set
// p (page) and s (shift) are variables used here
#define Vec_SetForEachEntry( Type, pVec, nSize, pSet, p, s )   \
    for ( p = 0; p <= pVec->iPage; p++ )                       \
98
        for ( s = 2; s < Vec_SetLimit(pVec->pPages[p]) && ((pSet) = (Type)(pVec->pPages[p] + (s))); s += nSize )
99 100 101 102 103 104 105 106 107 108

////////////////////////////////////////////////////////////////////////
///                     FUNCTION DEFINITIONS                         ///
////////////////////////////////////////////////////////////////////////

/**Function*************************************************************

  Synopsis    [Allocating vector.]

  Description []
109

110 111 112 113 114
  SideEffects []

  SeeAlso     []

***********************************************************************/
115
static inline void Vec_SetAlloc_( Vec_Set_t * p, int nPageSize )
116
{
117
    assert( nPageSize > 8 );
118
    memset( p, 0, sizeof(Vec_Set_t) );
119 120
    p->nPageSize    = nPageSize;
    p->uPageMask    = (unsigned)((1 << nPageSize) - 1);
121 122
    p->nPagesAlloc  = 256;
    p->pPages       = ABC_CALLOC( word *, p->nPagesAlloc );
123
    p->pPages[0]    = ABC_ALLOC( word, (1 << p->nPageSize) );
124
    p->pPages[0][0] = ~0;
125
    p->pPages[0][1] = ~0;
126
    Vec_SetWriteLimit( p->pPages[0], 2 );
127
}
128
static inline Vec_Set_t * Vec_SetAlloc( int nPageSize )
129 130 131
{
    Vec_Set_t * p;
    p = ABC_CALLOC( Vec_Set_t, 1 );
132
    Vec_SetAlloc_( p, nPageSize );
133 134 135 136 137 138 139 140
    return p;
}

/**Function*************************************************************

  Synopsis    [Resetting vector.]

  Description []
141

142 143 144 145 146 147 148 149 150 151 152
  SideEffects []

  SeeAlso     []

***********************************************************************/
static inline void Vec_SetRestart( Vec_Set_t * p )
{
    p->nEntries     = 0;
    p->iPage        = 0;
    p->iPageS       = 0;
    p->pPages[0][0] = ~0;
153
    p->pPages[0][1] = ~0;
154
    Vec_SetWriteLimit( p->pPages[0], 2 );
155 156 157 158
}

/**Function*************************************************************

159
  Synopsis    [Freeing vector.]
160 161 162 163 164 165 166 167

  Description []

  SideEffects []

  SeeAlso     []

***********************************************************************/
168
static inline void Vec_SetFree_( Vec_Set_t * p )
169
{
170
    int i;
171
    if ( p == NULL ) return;
172 173 174 175 176 177
    for ( i = 0; i < p->nPagesAlloc; i++ )
        ABC_FREE( p->pPages[i] );
    ABC_FREE( p->pPages );
}
static inline void Vec_SetFree( Vec_Set_t * p )
{
178
    if ( p == NULL ) return;
179 180
    Vec_SetFree_( p );
    ABC_FREE( p );
181 182 183 184
}

/**Function*************************************************************

185
  Synopsis    [Returns memory in bytes occupied by the vector.]
186 187

  Description []
188

189 190 191 192 193
  SideEffects []

  SeeAlso     []

***********************************************************************/
194
static inline double Vec_ReportMemory( Vec_Set_t * p )
195
{
196 197 198 199
    double Mem = sizeof(Vec_Set_t);
    Mem += p->nPagesAlloc * sizeof(void *);
    Mem += sizeof(word) * (1 << p->nPageSize) * (1 + p->iPage);
    return Mem;
200 201 202 203 204 205 206
}

/**Function*************************************************************

  Synopsis    [Appending entries to vector.]

  Description []
207

208 209 210 211 212 213 214
  SideEffects []

  SeeAlso     []

***********************************************************************/
static inline int Vec_SetAppend( Vec_Set_t * p, int * pArray, int nSize )
{
215
    int nWords = Vec_SetWordNum( nSize );
216 217
    assert( nWords < (1 << p->nPageSize) );
    p->nEntries++;
218
    if ( Vec_SetLimit( p->pPages[p->iPage] ) + nWords >= (1 << p->nPageSize) )
219 220 221 222 223 224 225 226
    {
        if ( ++p->iPage == p->nPagesAlloc )
        {
            p->pPages = ABC_REALLOC( word *, p->pPages, p->nPagesAlloc * 2 );
            memset( p->pPages + p->nPagesAlloc, 0, sizeof(word *) * p->nPagesAlloc );
            p->nPagesAlloc *= 2;
        }
        if ( p->pPages[p->iPage] == NULL )
227
            p->pPages[p->iPage] = ABC_ALLOC( word, (1 << p->nPageSize) );
228 229
        Vec_SetWriteLimit( p->pPages[p->iPage], 2 );
        p->pPages[p->iPage][1] = ~0;
230 231
    }
    if ( pArray )
232 233
        memcpy( p->pPages[p->iPage] + Vec_SetLimit(p->pPages[p->iPage]), pArray, sizeof(int) * nSize );
    Vec_SetIncLimit( p->pPages[p->iPage], nWords );
234 235 236 237
    return Vec_SetHandCurrent(p) - nWords;
}
static inline int Vec_SetAppendS( Vec_Set_t * p, int nSize )
{
238
    int nWords = Vec_SetWordNum( nSize );
239
    assert( nWords < (1 << p->nPageSize) );
240
    if ( Vec_SetLimitS( p->pPages[p->iPageS] ) + nWords >= (1 << p->nPageSize) )
241
        Vec_SetWriteLimitS( p->pPages[++p->iPageS], 2 );
242 243 244
    Vec_SetIncLimitS( p->pPages[p->iPageS], nWords );
    return Vec_SetHandCurrentS(p) - nWords;
}
245 246 247 248 249 250 251 252 253 254 255 256 257 258
static inline int Vec_SetFetchH( Vec_Set_t * p, int nBytes )
{
    return Vec_SetAppend(p, NULL, (nBytes + 3) >> 2);
}
static inline void * Vec_SetFetch( Vec_Set_t * p, int nBytes )
{
    return (void *)Vec_SetEntry( p, Vec_SetFetchH(p, nBytes) );
}
static inline char * Vec_SetStrsav( Vec_Set_t * p, char * pName )
{
    char * pStr = (char *)Vec_SetFetch( p, strlen(pName) + 1 );
    strcpy( pStr, pName );
    return pStr;
}
259 260 261 262 263 264

/**Function*************************************************************

  Synopsis    [Shrinking vector size.]

  Description []
265

266 267 268 269 270 271 272 273
  SideEffects [This procedure does not update the number of entries.]

  SeeAlso     []

***********************************************************************/
static inline void Vec_SetShrink( Vec_Set_t * p, int h )
{
    assert( h <= Vec_SetHandCurrent(p) );
274 275
    p->iPage = Vec_SetHandPage(p, h);
    Vec_SetWriteLimit( p->pPages[p->iPage], Vec_SetHandShift(p, h) );
276
}
277 278
static inline void Vec_SetShrinkS( Vec_Set_t * p, int h )
{
279
    assert( h <= Vec_SetHandCurrent(p) );
280 281
    p->iPageS = Vec_SetHandPage(p, h);
    Vec_SetWriteLimitS( p->pPages[p->iPageS], Vec_SetHandShift(p, h) );
282 283
}

284 285 286 287 288 289 290
static inline void Vec_SetShrinkLimits( Vec_Set_t * p )
{
    int i;
    for ( i = 0; i <= p->iPage; i++ )
        Vec_SetWriteLimit( p->pPages[i], Vec_SetLimitS(p->pPages[i]) );
}

291 292 293 294 295 296 297 298 299

ABC_NAMESPACE_HEADER_END

#endif

////////////////////////////////////////////////////////////////////////
///                       END OF FILE                                ///
////////////////////////////////////////////////////////////////////////