bionic (3) cdt.3.gz

Provided by: libgraphviz-dev_2.40.1-2_amd64 bug

NAME

       Cdt - container data types

SYNOPSIS

       #include <cdt.h>

   DICTIONARY TYPES
       Dt_t;
       Dtdisc_t;
       Dtmethod_t;
       Dtlink_t;
       Dtstat_t;

   DICTIONARY CONTROL
       Dt_t*       dtopen(const Dtdisc_t* disc, const Dtmethod_t* meth);
       int         dtclose(Dt_t* dt);
       void        dtclear(dt);
       Dtmethod_t* dtmethod(Dt_t* dt, const Dtmethod_t* meth);
       Dtdisc_t*   dtdisc(Dt_t* dt, const Dtdisc_t* disc, int type);
       Dt_t*       dtview(Dt_t* dt, Dt_t* view);
       int         dttreeset(Dt_t* dt, int minp, int balance);

   STORAGE METHODS
       Dtmethod_t* Dtset;
       Dtmethod_t* Dtbag;
       Dtmethod_t* Dtoset;
       Dtmethod_t* Dtobag;
       Dtmethod_t* Dtlist;
       Dtmethod_t* Dtstack;
       Dtmethod_t* Dtqueue;
       Dtmethod_t* Dtdeque;

   DISCIPLINE
       #define DTOFFSET(struct_s,member)
       #define DTDISC(disc,key,size,link,makef,freef,comparf,hashf,memoryf,eventf)
       typedef void*      (*Dtmake_f)(Dt_t*, void*, Dtdisc_t*);
       typedef void         (*Dtfree_f)(Dt_t*, void*, Dtdisc_t*);
       typedef int          (*Dtcompar_f)(Dt_t*, void*, void*, Dtdisc_t*);
       typedef unsigned int (*Dthash_f)(Dt_t*, void*, Dtdisc_t*);
       typedef void*      (*Dtmemory_f)(Dt_t*, void*, size_t, Dtdisc_t*);
       typedef int          (*Dtevent_f)(Dt_t*, int, void*, Dtdisc_t*);

   OBJECT OPERATIONS
       void*   dtinsert(Dt_t* dt, void* obj);
       void*   dtappend(Dt_t* dt, void* obj);
       void*   dtdelete(Dt_t* dt, void* obj);
       void*   dtattach(Dt_t* dt, void* obj);
       void*   dtdetach(Dt_t* dt, void* obj);
       void*   dtsearch(Dt_t* dt, void* obj);
       void*   dtmatch(Dt_t* dt, void* key);
       void*   dtfirst(Dt_t* dt);
       void*   dtnext(Dt_t* dt, void* obj);
       void*   dtlast(Dt_t* dt);
       void*   dtprev(Dt_t* dt, void* obj);
       void*   dtfinger(Dt_t* dt);
       void*   dtrenew(Dt_t* dt, void* obj);
       int       dtwalk(Dt_t* dt, int (*userf)(Dt_t*, void*, void*), void*);
       Dtlink_t* dtflatten(Dt_t* dt);
       Dtlink_t* dtlink(Dt_t*, Dtlink_t* link);
       void*   dtobj(Dt_t* dt, Dtlink_t* link);
       Dtlink_t* dtextract(Dt_t* dt);
       int       dtrestore(Dt_t* dt, Dtlink_t* link);

       #define   DTTREESEARCH(Dt_t* dt, void* obj, action)
       #define   DTTREEMATCH(Dt_t* dt, void* key, action)

   DICTIONARY STATUS
       Dt_t*     dtvnext(Dt_t* dt);
       int       dtvcount(Dt_t* dt);
       Dt_t*     dtvhere(Dt_t* dt);
       int       dtsize(Dt_t* dt);
       int       dtstat(Dt_t* dt, Dtstat_t*, int all);

   HASH FUNCTIONS
       unsigned int dtstrhash(unsigned int h, char* str, int n);
       unsigned int dtcharhash(unsigned int h, unsigned char c);

DESCRIPTION

       Cdt  manages  run-time  dictionaries using standard container data types: unordered set/multiset, ordered
       set/multiset, list, stack, and queue.

   DICTIONARY TYPES
     Dt_t
       This is the type of a dictionary handle.

     Dtdisc_t
       This defines the type  of  a  discipline  structure  which  describes  object  lay-out  and  manipulation
       functions.

     Dtmethod_t
       This defines the type of a container method.

     Dtlink_t
       This is the type of a dictionary object holder (see dtdisc().)

     Dtstat_t
       This is the type of a structure to return dictionary statistics (see dtstat().)

   DICTIONARY CONTROL
     Dt_t* dtopen(const Dtdisc_t* disc, const Dtmethod_t* meth)
       This creates a new dictionary.  disc is a discipline structure to describe object format.  meth specifies
       a manipulation method.  dtopen() returns the new dictionary or  NULL  on  error.   See  also  the  events
       DT_OPEN and DT_ENDOPEN below.

     int dtclose(Dt_t* dt)
       This  deletes  dt  and  its  objects.   Note  that  dtclose()  fails  if dt is being viewed by some other
       dictionaries (see dtview()).  dtclose() returns 0 on success and  -1  on  error.   See  also  the  events
       DT_CLOSE and DT_ENDCLOSE below.

     void dtclear(Dt_t* dt)
       This deletes all objects in dt without closing dt.

     Dtmethod_t dtmethod(Dt_t* dt, const Dtmethod_t* meth)
       If  meth  is NULL, dtmethod() returns the current method.  Otherwise, it changes the storage method of dt
       to meth.  Object order remains the same during  a  method  switch  among  Dtlist,  Dtstack,  Dtqueue  and
       Dtdeque.   Switching  to  and  from  Dtset/Dtbag  and  Dtoset/Dtobag  may  cause  objects to be rehashed,
       reordered, or removed as the case requires.  dtmethod() returns the previous method or NULL on error.

     Dtdisc_t* dtdisc(Dt_t* dt, const Dtdisc_t* disc, int type)
       If disc is NULL, dtdisc() returns the current discipline.  Otherwise, it changes the discipline of dt  to
       disc.  Objects may be rehashed, reordered, or removed as appropriate.  type can be any bit combination of
       DT_SAMECMP and DT_SAMEHASH.  DT_SAMECMP means that objects will compare exactly the same as  before  thus
       obviating  the  need  for  reordering  or removing new duplicates.  DT_SAMEHASH means that hash values of
       objects remain the same thus obviating the need to rehash.  dtdisc() returns the previous  discipline  on
       success and NULL on error.

     Dt_t* dtview(Dt_t* dt, Dt_t* view)
       A  viewpath  allows  a  search or walk starting from a dictionary to continue to another.  dtview() first
       terminates any current view from dt to another dictionary.  Then, if view is  NULL,  dtview  returns  the
       terminated  view  dictionary.   If view is not NULL, a viewpath from dt to view is established.  dtview()
       returns dt on success and NULL on error.

       It is an error to have  dictionaries  on  a  viewpath  with  different  storage  methods.   In  addition,
       dictionaries on the same view path should treat objects in a consistent manner with respect to comparison
       or hashing.  If not, undefined behaviors may result.

     int dttreeset(Dt_t* dt, int minp, int balance)
       This function only applies to dictionaries operated under the method Dtoset  which  uses  top-down  splay
       trees (see below). It returns 0 on success and -1 on error.

       minp:  This  parameter  defines  the  minimum path length before a search path is adjusted.  For example,
              minp equal 0 would mean that search paths are always adjusted.  If minp is negative,  the  minimum
              search  path  is  internally  computed  based  on  a function of the current dictionary size. This
              computed value is such that if the tree is balanced, it will never require adjusting.

       balance:
              If this is non-zero, the tree will be made balanced.

   STORAGE METHODS
       Storage methods are of type Dtmethod_t*.  Cdt supports the following methods:

     Dtoset
     Dtobag
       Objects are ordered by comparisons.  Dtoset keeps unique objects.  Dtobag allows repeatable objects.

     Dtset
     Dtbag
       Objects are unordered.  Dtset keeps unique objects.  Dtbag allows repeatable  objects  and  always  keeps
       them  together  (note the effect on dictionary walking.)  These methods use a hash table with chaining to
       manage the objects.  See also the event DT_HASHSIZE below on how  to  manage  hash  table  resizing  when
       objects are inserted.

     Dtlist
       Objects are kept in a list.  The call dtinsert() inserts a new object in front of the current object (see
       dtfinger()) if it is defined or at list front if no current  object  is  defined.   Similarly,  the  call
       dtappend() appends a new object after the current object (see dtfinger()) if it is defined or at list end
       if no current object is defined.

     Dtdeque
       Objects are kept in a deque. This is similar to Dtlist except that objects are  always  inserted  at  the
       front and appended at the tail of the list.

     Dtstack
       Objects  are  kept in a stack, i.e., in reverse order of insertion.  Thus, the last object inserted is at
       stack top and will be the first to be deleted.

     Dtqueue
       Objects are kept in a queue, i.e., in order of insertion.  Thus, the first object inserted  is  at  queue
       head and will be the first to be deleted.

   DISCIPLINE
       Object format and associated management functions are defined in the type Dtdisc_t:
           typedef struct
           { int        key, size;
             int        link;
             Dtmake_f   makef;
             Dtfree_f   freef;
             Dtcompar_f comparf;
             Dthash_f   hashf;
             Dtmemory_f memoryf;
             Dtevent_f  eventf;
           } Dtdisc_t;

     int key, size
       Each object obj is identified by a key used for object comparison or hashing.  key should be non-negative
       and defines an offset into obj.  If size is negative, the key is a null-terminated string  with  starting
       address  *(void**)((char*)obj+key).   If  size is zero, the key is a null-terminated string with starting
       address (void*)((char*)obj+key).  Finally, if size is positive, the key is a byte array  of  length  size
       starting at (void*)((char*)obj+key).

     int link
       Let  obj  be  an  object  to  be inserted into dt as discussed below.  If link is negative, an internally
       allocated object holder is used to hold obj. Otherwise, obj should have  a  Dtlink_t  structure  embedded
       link bytes into it, i.e., at address (Dtlink_t*)((char*)obj+link).

     void* (*makef)(Dt_t* dt, void* obj, Dtdisc_t* disc)
       If  makef  is  not  NULL,  dtinsert(dt,obj) or dtappend() will call it to make a copy of obj suitable for
       insertion into dt.  If makef is NULL, obj itself will be inserted into dt.

     void (*freef)(Dt_t* dt, void* obj, Dtdisc_t* disc)
       If not NULL, freef is used to destroy data associated with obj.

   int (*comparf)(Dt_t* dt, void* key1, void* key2, Dtdisc_t* disc)
       If not NULL, comparf is used to compare two keys.  Its return value should be <0, =0, or >0  to  indicate
       whether  key1  is  smaller,  equal  to, or larger than key2.  All three values are significant for method
       Dtoset and Dtobag.  For other methods, a zero value indicates equality and  a  non-zero  value  indicates
       inequality.   If (*comparf)() is NULL, an internal function is used to compare the keys as defined by the
       Dtdisc_t.size field.

     unsigned int (*hashf)(Dt_t* dt, void* key, Dtdisc_t* disc)
       If not NULL, hashf is used to compute the hash value of key.  It is required  that  keys  compared  equal
       will  also  have  same  hash  values.   If hashf is NULL, an internal function is used to hash the key as
       defined by the Dtdisc_t.size field.

     void* (*memoryf)(Dt_t* dt, void* addr, size_t size, Dtdisc_t* disc)
       If not NULL, memoryf is used to allocate and free memory.  When addr is NULL, a memory  segment  of  size
       size  is  requested.  If addr is not NULL and size is zero, addr is to be freed.  If addr is not NULL and
       size is positive, addr is to be resized to the given size.  If memoryf is NULL, malloc(3) is used.

     int (*eventf)(Dt_t* dt, int type, void* data, Dtdisc_t* disc)
       If not NULL, eventf announces various events.  Each event may have  particular  handling  of  the  return
       values from eventf.  But a negative return value typically means failure.  Following are the events:

       DT_OPEN:
              dt  is being opened.  If eventf returns negative, the opening process terminates with failure.  If
              eventf returns zero, the opening process proceeds in a default manner.  A  positive  return  value
              indicates special treatment of memory as follows.  If *(void**)data is set to point to some memory
              segment as discussed in memoryf, that segment of memory  is  used  to  start  the  dictionary.  If
              *(void**)data is NULL, all memory including that of the dictionary handle itself will be allocated
              via memoryf.

       DT_ENDOPEN:
              This event announces that dtopen() has successfully opened a dictionary and is  about  to  return.
              The data argument of eventf should be the new dictionary handle itself.

       DT_CLOSE:
              dt  is  about  to be closed. If eventf returns negative, the closing process stops immediately and
              dtclose() returns -1.  Objects in the dictionary are deleted only if  eventf  returns  zero.   The
              dictionary  handle  itself  is processed as follows.  If it was allocated via malloc(), it will be
              freed.  If it was allocated via memoryf (see dtopen()) and eventf returns 0,  a  call  to  memoryf
              will be issued to attempt freeing the handle.  Otherwise, nothing will be done to its memory.

              As should be clear from their description, the events DT_OPEN and DT_CLOSE are designed to be used
              along with memoryf to manage the allocation and  deallocation  of  dictionary  and  object  memory
              across  dictionaries.  In  fact,  they  can  be used to manage dictionaries based on shared and/or
              persistent memory.

       DT_ENDCLOSE:
              This event announces that dtclose() has successfully closed a dictionary and is about to return.

       DT_DISC:
              The discipline of dt is being changed to a new one given in (Dtdisc_t*)data.

       DT_METH:
              The method of dt is being changed to a new one given in (Dtmethod_t*)data.

       DT_HASHSIZE:
              The hash table (for Dtset and Dtbag) is being resized.  In this case, *(int*)data has the  current
              size  of  the  table.  The application can set the new table size by first changing *(int*)data to
              the desired size, then return a positive value.  The application can also fix the  table  size  at
              the current value forever by setting *(int*)data to a negative value, then again return a positive
              value. A non-positive return value from the  event  handling  function  means  that  Cdt  will  be
              responsible for choosing the hash table size.

   #define DTOFFSET(struct_s,member)
       This  macro function computes the offset of member from the start of structure struct_s. It is useful for
       getting the offset of a Dtlink_t embedded inside an object.

   #define DTDISC(disc,key,size,link,makef,freef,comparf,hashf,memoryf,eventf)
       This macro function initializes the discipline pointed to by disc with the given values.

   OBJECT OPERATIONS
     void* dtinsert(Dt_t* dt, void* obj)
     void* dtappend(Dt_t* dt, void* obj)
       These functions add an object prototyped by obj into dt.  dtinsert()  and  dtappend()  perform  the  same
       function for all methods except for Dtlist. See Dtlist for details.  If there is an existing object in dt
       matching obj and the storage method is Dtset or Dtoset, dtinsert() and dtappend() will simply return  the
       matching object.  Otherwise, a new object is inserted according to the method in use.  See Dtdisc_t.makef
       for object construction.  The new object or a matching object as noted will be returned on success  while
       NULL is returned on error.

     void* dtdelete(Dt_t* dt, void* obj)
       If  obj  is  NULL,  methods  Dtstack  and Dtqueue delete respectively stack top or queue head while other
       methods do nothing.  If obj is not NULL, there are two cases.  If the method  in  use  is  not  Dtbag  or
       Dtobag,  the  first  object matching obj is deleted.  On the other hand, if the method in use is Dtbag or
       Dtobag, the library check to see if obj is in the dictionary and  delete  it.   If  obj  is  not  in  the
       dictionary,  some  object  matching  it  will  be  deleted.   See  Dtdisc_t.freef for object destruction.
       dtdelete() returns the deleted object (even if it was deallocated) or NULL on error.

     void* dtattach(Dt_t* dt, void* obj)
       This function is similar to dtinsert() but obj itself will be inserted  into  dt  even  if  a  discipline
       function makef is defined.

     void* dtdetach(Dt_t* dt, void* obj)
       This  function  is  similar to dtdelete() but the object to be deleted from dt will not be freed (via the
       discipline freef function).

     void* dtsearch(Dt_t* dt, void* obj)
     void* dtmatch(Dt_t* dt, void* key)
       These functions find an object matching obj or key either from dt or from some dictionary accessible from
       dt  via  a  viewpath  (see  dtview().)   dtsearch()  and  dtmatch() return the matching object or NULL on
       failure.

     void* dtfirst(Dt_t* dt)
     void* dtnext(Dt_t* dt, void* obj)
       dtfirst() returns the first object in dt.  dtnext()  returns  the  object  following  obj.   Objects  are
       ordered  based  on  the  storage  method  in  use.   For Dtoset and Dtobag, objects are ordered by object
       comparisons.  For Dtstack, objects are ordered in reverse order of insertion.  For Dtqueue,  objects  are
       ordered  in  order of insertion.  For Dtlist, objects are ordered by list position.  For Dtset and Dtbag,
       objects are ordered by some internal order (more below).  Thus, objects in a dictionary or a viewpath can
       be walked using a for(;;) loop as below.
           for(obj = dtfirst(dt); obj; obj = dtnext(dt,obj))
       When  a dictionary uses Dtset or Dtbag, the object order is determined upon a call to dtfirst()/dtlast().
       This order is frozen until a call dtnext()/dtprev() returns NULL or when these same functions are  called
       with  a  NULL  object  argument.   It  is  important  that  a  dtfirst()/dtlast()  call  be balanced by a
       dtnext()/dtprev() call as described.  Nested loops will require multiple balancing, once  per  loop.   If
       loop balancing is not done carefully, either performance is degraded or unexpected behaviors may result.

     void* dtlast(Dt_t* dt)
     void* dtprev(Dt_t* dt, void* obj)
       dtlast()  and dtprev() are like dtfirst() and dtnext() but work in reverse order.  Note that dictionaries
       on a viewpath are still walked in order but objects in each dictionary are walked in reverse order.

     void* dtfinger(Dt_t* dt)
       This function returns the current object of dt, if any.  The current object is defined after a successful
       call  to one of dtsearch(), dtmatch(), dtinsert(), dtfirst(), dtnext(), dtlast(), or dtprev().  As a side
       effect of this implementation of Cdt, when a dictionary is based on Dtoset and Dtobag, the current object
       is always defined and is the root of the tree.

     void* dtrenew(Dt_t* dt, void* obj)
       This  function  repositions and perhaps rehashes an object obj after its key has been changed.  dtrenew()
       only works if obj is the current object (see dtfinger()).

     dtwalk(Dt_t* dt, int (*userf)(Dt_t*, void*, void*), void* data)
       This function calls (*userf)(walk,obj,data) on each object in dt and other dictionaries viewable from it.
       walk  is  the  dictionary containing obj.  If userf() returns a <0 value, dtwalk() terminates and returns
       the same value.  dtwalk() returns 0 on completion.

     Dtlink_t* dtflatten(Dt_t* dt)
     Dtlink_t* dtlink(Dt_t* dt, Dtlink_t* link)
     void* dtobj(Dt_t* dt, Dtlink_t* link)
       Using dtfirst()/dtnext() or dtlast()/dtprev() to walk a single dictionary can incur significant cost  due
       to  function  calls.  For efficient walking of a single directory (i.e., no viewpathing), dtflatten() and
       dtlink() can be used.  Objects in dt are made into a linked list and walked as follows:
           for(link = dtflatten(dt); link; link = dtlink(dt,link) )

       Note that dtflatten() returns a list of type Dtlink_t*, not void*.  That  is,  it  returns  a  dictionary
       holder  pointer,  not  a  user object pointer (although both are the same if the discipline field link is
       zero.)  The macro function dtlink() returns the dictionary  holder  object  following  link.   The  macro
       function  dtobj(dt,link)  returns  the user object associated with link, Beware that the flattened object
       list is unflattened on any dictionary operations other than dtlink().

     Dtlink_t* dtextract(Dt_t* dt)
     int dtrestore(Dt_t* dt, Dtlink_t* link)
       dtextract() extracts all objects from dt and makes it appear  empty.   dtrestore()  repopulates  dt  with
       objects  previously obtained via dtextract().  dtrestore() will fail if dt is not empty.  These functions
       can be used to share a same dt handle among many sets of objects.  They are useful to  reduce  dictionary
       overhead  in  an  application  that  creates many concurrent dictionaries.  It is important that the same
       discipline and method are in use at both extraction and restoration. Otherwise, undefined  behaviors  may
       result.

     #define   DTTREESEARCH(Dt_t* dt, void* obj, action)
     #define   DTTREEMATCH(Dt_t* dt, void* key, action)
       These macro functions are analogues of dtsearch() and dtmatch() but they can only be used on a dictionary
       based on a binary search tree, i.e., Dtoset or Dtobag.

       obj or key:
              These are used to find a matching object. If there is no match, the result is NULL.

       action:
              The matching object o (which may be NULL) will be processed as follow:

                  action (o);

              Since action is used verbatim, it can be any C  code  fragment  combinable  with  (o)  to  form  a
              syntactically  correct  C statement.  For example, suppose that the matching object is an integer,
              the below code accumulates the integer value in a variable total:

                  DTTREEMATCH(dt, key, total += (int));

   DICTIONARY INFORMATION
     Dt_t* dtvnext(Dt_t* dt)
       This returns the dictionary that dt is viewing, if any.

     int dtvcount(Dt_t* dt)
       This returns the number of dictionaries that view dt.

     Dt_t* dtvhere(Dt_t* dt)
       This returns the dictionary v viewable from dt where an object was found from the most recent  search  or
       walk operation.

     int dtsize(Dt_t* dt)
       This function returns the number of objects stored in dt.

     int dtstat(Dt_t *dt, Dtstat_t* st, int all)
       This  function  reports  dictionary  statistics.   If  all  is  non-zero,  all  fields  of st are filled.
       Otherwise, only the dt_type and dt_size fields are filled.  It returns 0 on success and -1 on error.

       Dtstat_t contains the below fields:

       int dt_type:
              This is one of DT_SET, DT_BAG, DT_OSET, DT_OBAG, DT_LIST, DT_STACK, and DT_QUEUE.

       int dt_size:
              This contains the number of objects in the dictionary.

       int dt_n:
              For Dtset and Dtbag, this is the number of non-empty chains in the hash  table.   For  Dtoset  and
              Dtobag,  this  is  the  deepest  level  in  the tree (counting from zero.)  Each level in the tree
              contains all nodes of equal distance from the root node.   dt_n  and  the  below  two  fields  are
              undefined for other methods.

       int dt_max:
              For Dtbag and Dtset, this is the size of a largest chain.  For Dtoset and Dtobag, this is the size
              of a largest level.

       int* dt_count:
              For Dtset and Dtbag, this is the list of counts for chains  of  particular  sizes.   For  example,
              dt_count[1]  is  the number of chains of size 1.  For Dtoset and Dtobag, this is the list of sizes
              of the levels.  For example, dt_count[1] is the size of level 1.

   HASH FUNCTIONS
     unsigned int dtcharhash(unsigned int h, char c)
     unsigned int dtstrhash(unsigned int h, char* str, int n)
       These functions compute hash values from bytes or strings.  dtcharhash() computes a new hash  value  from
       byte  c  and seed value h.  dtstrhash() computes a new hash value from string str and seed value h.  If n
       is positive, str is a byte array of length n; otherwise, str is a null-terminated string.

IMPLEMENTATION NOTES

       Dtset and Dtbag are based on hash tables with move-to-front collision  chains.   Dtoset  and  Dtobag  are
       based on top-down splay trees.  Dtlist, Dtstack and Dtqueue are based on doubly linked list.

AUTHOR

       Kiem-Phong Vo, kpv@research.att.com

                                                                                                       LIBCDT(3)