tskiplist.h 6.1 KB
Newer Older
H
hzcheng 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
/*
 * Copyright (c) 2019 TAOS Data, Inc. <jhtao@taosdata.com>
 *
 * This program is free software: you can use, redistribute, and/or modify
 * it under the terms of the GNU Affero General Public License, version 3
 * or later ("AGPL"), as published by the Free Software Foundation.
 *
 * 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.
 *
 * You should have received a copy of the GNU Affero General Public License
 * along with this program. If not, see <http://www.gnu.org/licenses/>.
 */

16 17
#ifndef TDENGINE_TSKIPLIST_H
#define TDENGINE_TSKIPLIST_H
H
hzcheng 已提交
18 19 20 21 22

#ifdef __cplusplus
extern "C" {
#endif

S
slguan 已提交
23
#include "os.h"
H
hzcheng 已提交
24
#include "taosdef.h"
H
more  
hzcheng 已提交
25
#include "tarray.h"
26
#include "tfunctional.h"
H
hzcheng 已提交
27

H
more  
hzcheng 已提交
28 29 30
#define MAX_SKIP_LIST_LEVEL 15
#define SKIP_LIST_RECORD_PERFORMANCE 0

H
TD-1437  
Hongze Cheng 已提交
31
// For key property setting
H
Hongze Cheng 已提交
32 33
#define SL_ALLOW_DUP_KEY (uint8_t)0x0    // Allow duplicate key exists (for tag index usage)
#define SL_DISCARD_DUP_KEY (uint8_t)0x1  // Discard duplicate key (for data update=0 case)
34 35
#define SL_UPDATE_DUP_KEY (uint8_t)0x2   // Update duplicate key by remove/insert (for data update!=0 case)

H
TD-1437  
Hongze Cheng 已提交
36 37 38
// For thread safety setting
#define SL_THREAD_SAFE (uint8_t)0x4

H
more  
hzcheng 已提交
39 40 41
typedef char *SSkipListKey;
typedef char *(*__sl_key_fn_t)(const void *);

42 43 44
typedef void (*sl_patch_row_fn_t)(void * pDst, const void * pSrc);
typedef void* (*iter_next_fn_t)(void *iter);

H
more  
hzcheng 已提交
45
typedef struct SSkipListNode {
H
TD-1194  
Hongze Cheng 已提交
46 47 48
  uint8_t        level;
  void *         pData;
  struct SSkipListNode *forwards[];
H
more  
hzcheng 已提交
49
} SSkipListNode;
H
hzcheng 已提交
50

H
TD-1194  
Hongze Cheng 已提交
51 52 53
#define SL_GET_NODE_DATA(n) (n)->pData
#define SL_NODE_GET_FORWARD_POINTER(n, l) (n)->forwards[(l)]
#define SL_NODE_GET_BACKWARD_POINTER(n, l) (n)->forwards[(n)->level + (l)]
H
hzcheng 已提交
54 55

/*
H
more  
hzcheng 已提交
56
 * @version 0.3
H
hzcheng 已提交
57
 * @date   2017/11/12
H
more  
hzcheng 已提交
58 59
 * the simple version of skip list.
 *
S
slguan 已提交
60
 * for multi-thread safe purpose, we employ pthread_rwlock_t to guarantee to generate
H
more  
hzcheng 已提交
61 62 63
 * deterministic result. Later, we will remove the lock in SkipList to further enhance the performance.
 * In this case, one should use the concurrent skip list (by using michael-scott algorithm) instead of
 * this simple version in a multi-thread environment, to achieve higher performance of read/write operations.
H
hzcheng 已提交
64 65
 *
 * Note: Duplicated primary key situation.
S
slguan 已提交
66
 * In case of duplicated primary key, two ways can be employed to handle this situation:
H
more  
hzcheng 已提交
67 68 69
 * 1. add as normal insertion without special process.
 * 2. add an overflow pointer at each list node, all nodes with the same key will be added in the overflow pointer.
 *    In this case, the total steps of each search will be reduced significantly.
S
slguan 已提交
70 71 72 73
 *    Currently, we implement the skip list in a line with the first means, maybe refactor it soon.
 *
 *    Memory consumption: the memory alignment causes many memory wasted. So, employ a memory
 *    pool will significantly reduce the total memory consumption, as well as the calloc/malloc operation costs.
H
hzcheng 已提交
74 75 76 77 78 79 80 81 82
 *
 */

// state struct, record following information:
// number of links in each level.
// avg search steps, for latest 1000 queries
// avg search rsp time, for latest 1000 queries
// total memory size
typedef struct tSkipListState {
H
more  
hzcheng 已提交
83
  // in bytes, sizeof(SSkipList)+sizeof(SSkipListNode)*SSkipList->nSize
H
hzcheng 已提交
84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102
  uint64_t nTotalMemSize;
  uint64_t nLevelNodeCnt[MAX_SKIP_LIST_LEVEL];
  uint64_t queryCount;  // total query count

  /*
   * only record latest 1000 queries
   * when the value==1000, = 0,
   * nTotalStepsForQueries = 0,
   * nTotalElapsedTimeForQueries = 0
   */
  uint64_t nRecQueries;
  uint16_t nTotalStepsForQueries;
  uint64_t nTotalElapsedTimeForQueries;

  uint16_t nInsertObjs;
  uint16_t nTotalStepsForInsert;
  uint64_t nTotalElapsedTimeForInsert;
} tSkipListState;

103 104 105 106 107 108
typedef enum {
  SSkipListPutSuccess    = 0,
  SSkipListPutEarlyStop  = 1,
  SSkipListPutSkipOne    = 2
} SSkipListPutStatus;

H
more  
hzcheng 已提交
109
typedef struct SSkipList {
110
  unsigned int      seed;
H
hjxilinx 已提交
111 112
  __compar_fn_t     comparFn;
  __sl_key_fn_t     keyFn;
H
TD-1437  
Hongze Cheng 已提交
113 114
  pthread_rwlock_t *lock;
  uint16_t          len;
H
hjxilinx 已提交
115
  uint8_t           maxLevel;
H
TD-1437  
Hongze Cheng 已提交
116
  uint8_t           flags;
H
TD-1194  
Hongze Cheng 已提交
117
  uint8_t           type;  // static info above
H
hjxilinx 已提交
118
  uint8_t           level;
H
TD-1548  
Hongze Cheng 已提交
119
  uint32_t          size;
H
TD-1437  
Hongze Cheng 已提交
120 121
  SSkipListNode *   pHead;  // point to the first element
  SSkipListNode *   pTail;  // point to the last element
H
more  
hzcheng 已提交
122 123 124
#if SKIP_LIST_RECORD_PERFORMANCE
  tSkipListState state;  // skiplist state
#endif
125
  tGenericSavedFunc* insertHandleFn;
H
more  
hzcheng 已提交
126
} SSkipList;
H
hzcheng 已提交
127

S
slguan 已提交
128
typedef struct SSkipListIterator {
H
more  
hzcheng 已提交
129 130
  SSkipList *    pSkipList;
  SSkipListNode *cur;
131 132
  int32_t        step;          // the number of nodes that have been checked already
  int32_t        order;         // order of the iterator
133
  SSkipListNode *next;          // next points to the true qualified node in skiplist
S
slguan 已提交
134 135
} SSkipListIterator;

H
TD-1194  
Hongze Cheng 已提交
136 137 138 139 140 141 142
#define SL_IS_THREAD_SAFE(s) (((s)->flags) & SL_THREAD_SAFE)
#define SL_DUP_MODE(s) (((s)->flags) & ((((uint8_t)1) << 2) - 1))
#define SL_GET_NODE_KEY(s, n) ((s)->keyFn((n)->pData))
#define SL_GET_MIN_KEY(s) SL_GET_NODE_KEY(s, SL_NODE_GET_FORWARD_POINTER((s)->pHead, 0))
#define SL_GET_MAX_KEY(s) SL_GET_NODE_KEY((s), SL_NODE_GET_BACKWARD_POINTER((s)->pTail, 0))
#define SL_SIZE(s) (s)->size

H
TD-1548  
Hongze Cheng 已提交
143 144 145 146
SSkipList *tSkipListCreate(uint8_t maxLevel, uint8_t keyType, uint16_t keyLen, __compar_fn_t comparFn, uint8_t flags,
                           __sl_key_fn_t fn);
void       tSkipListDestroy(SSkipList *pSkipList);
SSkipListNode *    tSkipListPut(SSkipList *pSkipList, void *pData);
147
void               tSkipListPutBatchByIter(SSkipList *pSkipList, void *iter, iter_next_fn_t iterate);
H
TD-1548  
Hongze Cheng 已提交
148 149
SArray *           tSkipListGet(SSkipList *pSkipList, SSkipListKey pKey);
void               tSkipListPrint(SSkipList *pSkipList, int16_t nlevel);
H
hjxilinx 已提交
150
SSkipListIterator *tSkipListCreateIter(SSkipList *pSkipList);
H
TD-1194  
Hongze Cheng 已提交
151 152 153 154 155 156
SSkipListIterator *tSkipListCreateIterFromVal(SSkipList *pSkipList, const char *val, int32_t type, int32_t order);
bool               tSkipListIterNext(SSkipListIterator *iter);
SSkipListNode *    tSkipListIterGet(SSkipListIterator *iter);
void *             tSkipListDestroyIter(SSkipListIterator *iter);
uint32_t           tSkipListRemove(SSkipList *pSkipList, SSkipListKey key);
void               tSkipListRemoveNode(SSkipList *pSkipList, SSkipListNode *pNode);
H
hzcheng 已提交
157 158 159 160 161

#ifdef __cplusplus
}
#endif

162
#endif  // TDENGINE_TSKIPLIST_H