• Home
  • Features
  • Pricing
  • Docs
  • Announcements
  • Sign In

taosdata / TDengine / #4759

25 Sep 2025 05:58AM UTC coverage: 58.866% (+0.03%) from 58.832%
#4759

push

travis-ci

web-flow
enh: taos command line support '-uroot' on windows (#33055)

135700 of 293169 branches covered (46.29%)

Branch coverage included in aggregate %.

204479 of 284720 relevant lines covered (71.82%)

25399510.59 hits per line

Source File
Press 'n' to go to next uncovered line, 'b' for previous

45.17
/source/common/src/tdataformat.c
1
/*
2
 * Copyright (c) 2019 TAOS Data, Inc. <jhtao@taosdata.com>
3
 *
4
 * This program is free software: you can use, redistribute, and/or modify
5
 * it under the terms of the GNU Affero General Public License, version 3
6
 * or later ("AGPL"), as published by the Free Software Foundation.
7
 *
8
 * This program is distributed in the hope that it will be useful, but WITHOUT
9
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
10
 * FITNESS FOR A PARTICULAR PURPOSE.
11
 *
12
 * You should have received a copy of the GNU Affero General Public License
13
 * along with this program. If not, see <http://www.gnu.org/licenses/>.
14
 */
15

16
#define _DEFAULT_SOURCE
17
#include "tdataformat.h"
18
#include "decimal.h"
19
#include "tRealloc.h"
20
#include "tdatablock.h"
21
#include "tlog.h"
22

23
enum {
24
  BLOB_ROW_KV = 0,
25
  BLOB_ROW_TUPLE = 1,
26
};
27

28
static int32_t (*tColDataAppendValueImpl[8][3])(SColData *pColData, uint8_t *pData, uint32_t nData);
29
static int32_t (*tColDataUpdateValueImpl[8][3])(SColData *pColData, uint8_t *pData, uint32_t nData, bool forward);
30

31
static int32_t tRowMergeWithBlobImpl(SArray *aRowP, STSchema *pTSchema, SBlobSet *pBlob, SBlobSet *pDstBlobSet,
32
                                     int32_t iStart, int32_t iEnd, int8_t flag);
33

34
static int32_t tRowBuildTupleWithBlob2(SArray *aColVal, const SRowBuildScanInfo *sinfo, const STSchema *schema,
35
                                       SRow **ppRow, SBlobSet *pSrcBlobSet, SBlobSet *pDstBlobSet);
36

37
static int32_t tRowBuildKVRowWithBlob2(SArray *aColVal, const SRowBuildScanInfo *sinfo, const STSchema *schema,
38
                                       SRow **ppRow, SBlobSet *pSrcBlob, SBlobSet *pDstBlobSet);
39
// ================================
40
static int32_t tGetTagVal(uint8_t *p, STagVal *pTagVal, int8_t isJson);
41

42
// SRow ========================================================================
43
#define KV_FLG_LIT ((uint8_t)0x10)
44
#define KV_FLG_MID ((uint8_t)0x20)
45
#define KV_FLG_BIG ((uint8_t)0x40)
46

47
#define BIT_FLG_NONE  ((uint8_t)0x0)
48
#define BIT_FLG_NULL  ((uint8_t)0x1)
49
#define BIT_FLG_VALUE ((uint8_t)0x2)
50

51
#pragma pack(push, 1)
52
typedef struct {
53
  int16_t nCol;
54
  uint8_t idx[];  // uint8_t * | uint16_t * | uint32_t *
55
} SKVIdx;
56
#pragma pack(pop)
57

58
#define ROW_SET_BITMAP(PB, FLAG, IDX, VAL)      \
59
  do {                                          \
60
    switch (FLAG) {                             \
61
      case (HAS_NULL | HAS_NONE):               \
62
        SET_BIT1(PB, IDX, VAL);                 \
63
        break;                                  \
64
      case (HAS_VALUE | HAS_NONE):              \
65
        SET_BIT1(PB, IDX, (VAL) ? (VAL)-1 : 0); \
66
        break;                                  \
67
      case (HAS_VALUE | HAS_NULL):              \
68
        SET_BIT1(PB, IDX, (VAL)-1);             \
69
        break;                                  \
70
      case (HAS_VALUE | HAS_NULL | HAS_NONE):   \
71
        SET_BIT2(PB, IDX, VAL);                 \
72
        break;                                  \
73
      default:                                  \
74
        break;                                  \
75
    }                                           \
76
  } while (0)
77

78
static int32_t tPutPrimaryKeyIndex(uint8_t *p, const SPrimaryKeyIndex *index) {
1,246,496,512✔
79
  int32_t n = 0;
1,246,496,512✔
80
  n += tPutI8(p ? p + n : p, index->type);
1,246,496,512✔
81
  n += tPutU32v(p ? p + n : p, index->offset);
1,246,496,512✔
82
  return n;
1,246,496,512✔
83
}
84

85
static int32_t tGetPrimaryKeyIndex(uint8_t *p, SPrimaryKeyIndex *index) {
2,147,483,647✔
86
  int32_t n = 0;
2,147,483,647✔
87
  n += tGetI8(p + n, &index->type);
2,147,483,647!
88
  n += tGetU32v(p + n, &index->offset);
2,147,483,647!
89
  return n;
2,147,483,647✔
90
}
91

92
static FORCE_INLINE int32_t tRowBuildScanAddNone(SRowBuildScanInfo *sinfo, const STColumn *pTColumn) {
93
  if ((pTColumn->flags & COL_IS_KEY)) return TSDB_CODE_PAR_PRIMARY_KEY_IS_NONE;
×
94
  sinfo->numOfNone++;
2,147,483,647✔
95
  return 0;
2,147,483,647✔
96
}
97

98
static FORCE_INLINE int32_t tRowBuildScanAddNull(SRowBuildScanInfo *sinfo, const STColumn *pTColumn) {
99
  if ((pTColumn->flags & COL_IS_KEY)) return TSDB_CODE_PAR_PRIMARY_KEY_IS_NULL;
12✔
100
  sinfo->numOfNull++;
15,843,070✔
101
  sinfo->kvMaxOffset = sinfo->kvPayloadSize;
15,843,070✔
102
  sinfo->kvPayloadSize += tPutI16v(NULL, -pTColumn->colId);
15,843,070✔
103
  return 0;
15,843,070✔
104
}
105

106
static FORCE_INLINE void tRowBuildScanAddValue(SRowBuildScanInfo *sinfo, SColVal *colVal, const STColumn *pTColumn) {
107
  bool isPK = ((pTColumn->flags & COL_IS_KEY) != 0);
2,147,483,647✔
108

109
  if (isPK) {
2,147,483,647✔
110
    sinfo->tupleIndices[sinfo->numOfPKs].type = colVal->value.type;
422,989,063✔
111
    sinfo->tupleIndices[sinfo->numOfPKs].offset =
422,989,063✔
112
        IS_VAR_DATA_TYPE(pTColumn->type) ? sinfo->tupleVarSize + sinfo->tupleFixedSize : pTColumn->offset;
422,989,063!
113
    sinfo->kvIndices[sinfo->numOfPKs].type = colVal->value.type;
422,989,063✔
114
    sinfo->kvIndices[sinfo->numOfPKs].offset = sinfo->kvPayloadSize;
422,989,063✔
115
    sinfo->numOfPKs++;
422,989,063✔
116
  }
117

118
  sinfo->kvMaxOffset = sinfo->kvPayloadSize;
2,147,483,647✔
119
  if (IS_VAR_DATA_TYPE(colVal->value.type)) {
2,147,483,647!
120
    if (sinfo->hasBlob && IS_STR_DATA_BLOB(colVal->value.type)) {
907,716,162!
121
      sinfo->tupleVarSize += tPutU32v(NULL, colVal->value.nData)     // size
×
122
                             + BSE_SEQUECE_SIZE;                     // value
×
123
      sinfo->kvPayloadSize += tPutI16v(NULL, colVal->cid)            // colId
×
124
                              + tPutU32v(NULL, colVal->value.nData)  // size
×
125
                              + BSE_SEQUECE_SIZE;                    // seq offset
×
126
    } else {
127
      sinfo->tupleVarSize += tPutU32v(NULL, colVal->value.nData)  // size
907,716,162✔
128
                             + colVal->value.nData;               // value
907,716,162✔
129

130
      sinfo->kvPayloadSize += tPutI16v(NULL, colVal->cid)            // colId
907,716,162✔
131
                              + tPutU32v(NULL, colVal->value.nData)  // size
907,716,162✔
132
                              + colVal->value.nData;                 // value
907,716,162✔
133
    }
134
  } else {
135
    sinfo->kvPayloadSize += tPutI16v(NULL, colVal->cid)              // colId
2,147,483,647✔
136
                            + tDataTypes[colVal->value.type].bytes;  // value
2,147,483,647✔
137
  }
138
  sinfo->numOfValue++;
2,147,483,647✔
139
}
2,147,483,647✔
140

141
static int32_t tRowBuildScan(SArray *colVals, const STSchema *schema, SRowBuildScanInfo *sinfo) {
2,129,629,675✔
142
  int32_t  code = 0;
2,129,629,675✔
143
  int32_t  colValIndex = 1;
2,129,629,675✔
144
  int32_t  numOfColVals = TARRAY_SIZE(colVals);
2,129,629,675✔
145
  SColVal *colValArray = (SColVal *)TARRAY_DATA(colVals);
2,129,629,675✔
146

147
  if (!(numOfColVals > 0)) {
2,129,629,675!
148
    return TSDB_CODE_INVALID_PARA;
×
149
  }
150
  if (!(colValArray[0].cid == PRIMARYKEY_TIMESTAMP_COL_ID)) {
2,129,629,675!
151
    return TSDB_CODE_PAR_INVALID_FIRST_COLUMN;
×
152
  }
153
  if (!(colValArray[0].value.type == TSDB_DATA_TYPE_TIMESTAMP)) {
2,129,629,675!
154
    return TSDB_CODE_PAR_INVALID_FIRST_COLUMN;
×
155
  }
156

157
  *sinfo = (SRowBuildScanInfo){.tupleFixedSize = schema->flen, .hasBlob = sinfo->hasBlob, .scanType = sinfo->scanType};
2,129,629,675✔
158

159
  // loop scan
160
  for (int32_t i = 1; i < schema->numOfCols; i++) {
2,147,483,647✔
161
    for (;;) {
162
      if (colValIndex >= numOfColVals) {
2,147,483,647✔
163
        if ((code = tRowBuildScanAddNone(sinfo, schema->columns + i))) goto _exit;
644!
164
        break;
322✔
165
      }
166

167
      if (colValArray[colValIndex].cid == schema->columns[i].colId) {
2,147,483,647✔
168
        if (!(colValArray[colValIndex].value.type == schema->columns[i].type)) {
2,147,483,647!
169
          code = TSDB_CODE_INVALID_PARA;
×
170
          goto _exit;
×
171
        }
172

173
        if (COL_VAL_IS_VALUE(&colValArray[colValIndex])) {
2,147,483,647✔
174
          tRowBuildScanAddValue(sinfo, &colValArray[colValIndex], schema->columns + i);
2,147,483,647✔
175
        } else if (COL_VAL_IS_NULL(&colValArray[colValIndex])) {
2,147,483,647✔
176
          if ((code = tRowBuildScanAddNull(sinfo, schema->columns + i))) goto _exit;
31,686,164✔
177
        } else if (COL_VAL_IS_NONE(&colValArray[colValIndex])) {
2,147,483,647!
178
          if ((code = tRowBuildScanAddNone(sinfo, schema->columns + i))) goto _exit;
2,147,483,647!
179
        }
180

181
        colValIndex++;
2,147,483,647✔
182
        break;
2,147,483,647✔
183
      } else if (colValArray[colValIndex].cid > schema->columns[i].colId) {
638,889✔
184
        if ((code = tRowBuildScanAddNone(sinfo, schema->columns + i))) goto _exit;
464!
185
        break;
232✔
186
      } else {  // skip useless value
187
        colValIndex++;
638,657✔
188
      }
189
    }
190
  }
191

192
  if (sinfo->numOfNone) {
2,129,629,663✔
193
    sinfo->flag |= HAS_NONE;
653,697,150✔
194
  }
195
  if (sinfo->numOfNull) {
2,129,629,663✔
196
    sinfo->flag |= HAS_NULL;
14,901,493✔
197
  }
198
  if (sinfo->numOfValue) {
2,129,629,663✔
199
    sinfo->flag |= HAS_VALUE;
2,087,325,306✔
200
  }
201

202
  // Tuple
203
  sinfo->tupleFlag = sinfo->flag;
2,129,629,663✔
204
  switch (sinfo->flag) {
2,129,629,663!
205
    case HAS_NONE:
46,859,946✔
206
    case HAS_NULL:
207
      sinfo->tupleBitmapSize = 0;
46,859,946✔
208
      sinfo->tupleFixedSize = 0;
46,859,946✔
209
      break;
46,859,946✔
210
    case HAS_VALUE:
1,468,897,389✔
211
      sinfo->tupleBitmapSize = 0;
1,468,897,389✔
212
      sinfo->tupleFixedSize = schema->flen;
1,468,897,389✔
213
      break;
1,468,897,389✔
214
    case (HAS_NONE | HAS_NULL):
23,075✔
215
      sinfo->tupleBitmapSize = BIT1_SIZE(schema->numOfCols - 1);
23,075✔
216
      sinfo->tupleFixedSize = 0;
23,075✔
217
      break;
23,075✔
218
    case (HAS_NONE | HAS_VALUE):
620,882,637✔
219
    case (HAS_NULL | HAS_VALUE):
220
      sinfo->tupleBitmapSize = BIT1_SIZE(schema->numOfCols - 1);
620,882,637✔
221
      sinfo->tupleFixedSize = schema->flen;
620,882,637✔
222
      break;
620,882,637✔
223
    case (HAS_NONE | HAS_NULL | HAS_VALUE):
428,527✔
224
      sinfo->tupleBitmapSize = BIT2_SIZE(schema->numOfCols - 1);
428,527✔
225
      sinfo->tupleFixedSize = schema->flen;
428,527✔
226
      break;
428,527✔
227
  }
228
  for (int32_t i = 0; i < sinfo->numOfPKs; i++) {
2,147,483,647✔
229
    sinfo->tupleIndices[i].offset += sinfo->tupleBitmapSize;
420,630,542✔
230
    sinfo->tuplePKSize += tPutPrimaryKeyIndex(NULL, sinfo->tupleIndices + i);
420,630,542✔
231
  }
232
  sinfo->tupleRowSize = sizeof(SRow)              // SRow
2,129,006,121✔
233
                        + sinfo->tuplePKSize      // primary keys
2,129,006,121✔
234
                        + sinfo->tupleBitmapSize  // bitmap
2,129,006,121✔
235
                        + sinfo->tupleFixedSize   // fixed part
2,129,006,121✔
236
                        + sinfo->tupleVarSize;    // var part
2,129,006,121✔
237

238
  // Key-Value
239
  if (sinfo->kvMaxOffset <= UINT8_MAX) {
2,129,006,121!
240
    sinfo->kvFlag = (KV_FLG_LIT | sinfo->flag);
2,132,355,908✔
241
    sinfo->kvIndexSize = sizeof(SKVIdx) + (sinfo->numOfNull + sinfo->numOfValue) * sizeof(uint8_t);
2,132,355,908✔
242
  } else if (sinfo->kvMaxOffset <= UINT16_MAX) {
×
243
    sinfo->kvFlag = (KV_FLG_MID | sinfo->flag);
4,831,170✔
244
    sinfo->kvIndexSize = sizeof(SKVIdx) + (sinfo->numOfNull + sinfo->numOfValue) * sizeof(uint16_t);
4,831,170✔
245
  } else {
246
    sinfo->kvFlag = (KV_FLG_BIG | sinfo->flag);
×
247
    sinfo->kvIndexSize = sizeof(SKVIdx) + (sinfo->numOfNull + sinfo->numOfValue) * sizeof(uint32_t);
×
248
  }
249
  for (int32_t i = 0; i < sinfo->numOfPKs; i++) {
2,147,483,647✔
250
    sinfo->kvIndices[i].offset += sinfo->kvIndexSize;
420,606,705✔
251
    sinfo->kvPKSize += tPutPrimaryKeyIndex(NULL, sinfo->kvIndices + i);
420,606,705✔
252
  }
253
  sinfo->kvRowSize = sizeof(SRow)             // SRow
2,127,802,107✔
254
                     + sinfo->kvPKSize        // primary keys
2,127,802,107✔
255
                     + sinfo->kvIndexSize     // index array
2,127,802,107✔
256
                     + sinfo->kvPayloadSize;  // payload
2,127,802,107✔
257

258
_exit:
2,127,802,119✔
259
  return code;
2,127,802,119✔
260
}
261

262
static int32_t tRowBuildTupleRow(SArray *aColVal, const SRowBuildScanInfo *sinfo, const STSchema *schema,
1,527,130,317✔
263
                                 SRow **ppRow) {
264
  SColVal *colValArray = (SColVal *)TARRAY_DATA(aColVal);
1,527,130,317✔
265
  *ppRow = (SRow *)taosMemoryCalloc(1, sinfo->tupleRowSize);
1,527,130,317!
266
  if (*ppRow == NULL) {
1,539,076,451!
267
    return terrno;
×
268
  }
269
  (*ppRow)->flag = sinfo->tupleFlag;
1,539,076,451✔
270
  (*ppRow)->numOfPKs = sinfo->numOfPKs;
1,539,076,451✔
271
  (*ppRow)->sver = schema->version;
1,539,076,451✔
272
  (*ppRow)->len = sinfo->tupleRowSize;
1,539,076,451✔
273
  (*ppRow)->ts = VALUE_GET_TRIVIAL_DATUM(&colValArray[0].value);
1,539,076,451✔
274

275
  if (sinfo->tupleFlag == HAS_NONE || sinfo->tupleFlag == HAS_NULL) {
1,539,076,451✔
276
    return 0;
49,269,534✔
277
  }
278

279
  uint8_t *primaryKeys = (*ppRow)->data;
1,489,806,917✔
280
  uint8_t *bitmap = primaryKeys + sinfo->tuplePKSize;
1,489,806,917✔
281
  uint8_t *fixed = bitmap + sinfo->tupleBitmapSize;
1,489,806,917✔
282
  uint8_t *varlen = fixed + sinfo->tupleFixedSize;
1,489,806,917✔
283

284
  // primary keys
285
  for (int32_t i = 0; i < sinfo->numOfPKs; i++) {
1,913,573,142✔
286
    primaryKeys += tPutPrimaryKeyIndex(primaryKeys, sinfo->tupleIndices + i);
426,951,189✔
287
  }
288

289
  // bitmap + fixed + varlen
290
  int32_t numOfColVals = TARRAY_SIZE(aColVal);
1,486,621,953✔
291
  int32_t colValIndex = 1;
1,486,621,953✔
292
  for (int32_t i = 1; i < schema->numOfCols; i++) {
2,147,483,647✔
293
    for (;;) {
294
      if (colValIndex >= numOfColVals) {  // NONE
2,147,483,647✔
295
        ROW_SET_BITMAP(bitmap, sinfo->tupleFlag, i - 1, BIT_FLG_NONE);
22!
296
        break;
22✔
297
      }
298

299
      if (colValArray[colValIndex].cid == schema->columns[i].colId) {
2,147,483,647✔
300
        if (COL_VAL_IS_VALUE(&colValArray[colValIndex])) {  // value
2,147,483,647✔
301
          ROW_SET_BITMAP(bitmap, sinfo->tupleFlag, i - 1, BIT_FLG_VALUE);
2,147,483,647!
302

303
          if (IS_VAR_DATA_TYPE(schema->columns[i].type)) {
2,147,483,647!
304
            *(int32_t *)(fixed + schema->columns[i].offset) = varlen - fixed - sinfo->tupleFixedSize;
72,614,622✔
305
            varlen += tPutU32v(varlen, colValArray[colValIndex].value.nData);
72,614,622✔
306
            if (colValArray[colValIndex].value.nData) {
72,614,622!
307
              (void)memcpy(varlen, colValArray[colValIndex].value.pData, colValArray[colValIndex].value.nData);
72,960,133✔
308
              varlen += colValArray[colValIndex].value.nData;
72,960,133✔
309
            }
310
          } else {
311
            (void)memcpy(fixed + schema->columns[i].offset,
2,147,483,647✔
312
                         VALUE_GET_DATUM(&colValArray[colValIndex].value, schema->columns[i].type),
2,147,483,647!
313
                         tDataTypes[schema->columns[i].type].bytes);
2,147,483,647✔
314
          }
315
        } else if (COL_VAL_IS_NULL(&colValArray[colValIndex])) {  // NULL
13,090,576!
316
          ROW_SET_BITMAP(bitmap, sinfo->tupleFlag, i - 1, BIT_FLG_NULL);
14,338,323!
317
        } else if (COL_VAL_IS_NONE(&colValArray[colValIndex])) {  // NONE
×
318
          ROW_SET_BITMAP(bitmap, sinfo->tupleFlag, i - 1, BIT_FLG_NONE);
426,748!
319
        }
320

321
        colValIndex++;
2,147,483,647✔
322
        break;
2,147,483,647✔
323
      } else if (colValArray[colValIndex].cid > schema->columns[i].colId) {  // NONE
543,974✔
324
        ROW_SET_BITMAP(bitmap, sinfo->tupleFlag, i - 1, BIT_FLG_NONE);
4!
325
        break;
4✔
326
      } else {
327
        colValIndex++;
543,970✔
328
      }
329
    }
330
  }
331

332
  return 0;
1,486,621,953✔
333
}
334

335
static int32_t addEmptyItemToBlobSet(SBlobSet *pBlobSet, int8_t type, uint64_t *pSeq) {
×
336
  SBlobItem item = {.type = type, .len = 0};
×
337
  uint64_t  seq = 0;
×
338
  int32_t   code = tBlobSetPush(pBlobSet, &item, &seq, 0);
×
339
  if (code != 0) return code;
×
340

341
  if (pSeq != NULL) {
×
342
    *pSeq = seq;
×
343
  }
344
  return 0;
×
345
}
346
static int32_t tRowBuildTupleWithBlob(SArray *aColVal, const SRowBuildScanInfo *sinfo, const STSchema *schema,
×
347
                                      SRow **ppRow, SBlobSet *pBlobSet) {
348
  int32_t  code = 0;
×
349
  SColVal *colValArray = (SColVal *)TARRAY_DATA(aColVal);
×
350
  *ppRow = (SRow *)taosMemoryCalloc(1, sinfo->tupleRowSize);
×
351
  if (*ppRow == NULL) {
×
352
    return terrno;
×
353
  }
354
  (*ppRow)->flag = sinfo->tupleFlag;
×
355
  (*ppRow)->numOfPKs = sinfo->numOfPKs;
×
356
  (*ppRow)->sver = schema->version;
×
357
  (*ppRow)->len = sinfo->tupleRowSize;
×
358
  (*ppRow)->ts = colValArray[0].value.val;
×
359

360
  if (((*ppRow)->flag) == 0) {
×
361
    return TSDB_CODE_INVALID_PARA;
×
362
  }
363
  if (sinfo->tupleFlag == HAS_NONE || sinfo->tupleFlag == HAS_NULL) {
×
364
    code = addEmptyItemToBlobSet(pBlobSet, TSDB_DATA_BLOB_NULL_VALUE, NULL);
×
365
    return code;
×
366
  }
367

368
  uint8_t *primaryKeys = (*ppRow)->data;
×
369
  uint8_t *bitmap = primaryKeys + sinfo->tuplePKSize;
×
370
  uint8_t *fixed = bitmap + sinfo->tupleBitmapSize;
×
371
  uint8_t *varlen = fixed + sinfo->tupleFixedSize;
×
372
  uint8_t *start = varlen;
×
373

374
  // primary keys
375
  for (int32_t i = 0; i < sinfo->numOfPKs; i++) {
×
376
    primaryKeys += tPutPrimaryKeyIndex(primaryKeys, sinfo->tupleIndices + i);
×
377
  }
378

379
  // bitmap + fixed + varlen
380
  int32_t numOfColVals = TARRAY_SIZE(aColVal);
×
381
  int32_t colValIndex = 1;
×
382
  int8_t  firstBlobCol = 1;
×
383
  for (int32_t i = 1; i < schema->numOfCols; i++) {
×
384
    for (;;) {
×
385
      if (colValIndex >= numOfColVals) {  // NONE
×
386
        ROW_SET_BITMAP(bitmap, sinfo->tupleFlag, i - 1, BIT_FLG_NONE);
×
387
        break;
×
388
      }
389

390
      int8_t hasBlob = 0;
×
391
      if (colValArray[colValIndex].cid == schema->columns[i].colId) {
×
392
        if (COL_VAL_IS_VALUE(&colValArray[colValIndex])) {  // value
×
393
          ROW_SET_BITMAP(bitmap, sinfo->tupleFlag, i - 1, BIT_FLG_VALUE);
×
394

395
          if (IS_VAR_DATA_TYPE(schema->columns[i].type)) {
×
396
            if (IS_STR_DATA_BLOB(schema->columns[i].type)) {
×
397
              hasBlob = 1;
×
398
            }
399
            *(int32_t *)(fixed + schema->columns[i].offset) = varlen - fixed - sinfo->tupleFixedSize;
×
400
            varlen += tPutU32v(varlen, colValArray[colValIndex].value.nData);
×
401
            if (colValArray[colValIndex].value.nData) {
×
402
              if (hasBlob == 0) {
×
403
                (void)memcpy(varlen, colValArray[colValIndex].value.pData, colValArray[colValIndex].value.nData);
×
404
                varlen += colValArray[colValIndex].value.nData;
×
405
              } else {
406
                uint64_t  seq = 0;
×
407
                SBlobItem item = {.seqOffsetInRow = varlen - (*ppRow)->data,
×
408
                                  .data = colValArray[colValIndex].value.pData,
×
409
                                  .len = colValArray[colValIndex].value.nData,
×
410
                                  .type = TSDB_DATA_BLOB_VALUE};
411
                code = tBlobSetPush(pBlobSet, &item, &seq, firstBlobCol);
×
412
                if (firstBlobCol == 1) {
×
413
                  firstBlobCol = 0;
×
414
                }
415
                if (code != 0) return code;
×
416
                varlen += tPutU64(varlen, seq);
×
417
              }
418
            } else {
419
              if (hasBlob) TAOS_CHECK_RETURN(addEmptyItemToBlobSet(pBlobSet, TSDB_DATA_BLOB_EMPTY_VALUE, NULL));
×
420
            }
421
          } else {
422
            (void)memcpy(fixed + schema->columns[i].offset,
×
423
                         VALUE_GET_DATUM(&colValArray[colValIndex].value, schema->columns[i].type),
×
424
                         tDataTypes[schema->columns[i].type].bytes);
×
425
          }
426
        } else if (COL_VAL_IS_NULL(&colValArray[colValIndex])) {  // NULL
×
427
          ROW_SET_BITMAP(bitmap, sinfo->tupleFlag, i - 1, BIT_FLG_NULL);
×
428
          if (IS_STR_DATA_BLOB(schema->columns[i].type)) {
×
429
            TAOS_CHECK_RETURN(addEmptyItemToBlobSet(pBlobSet, TSDB_DATA_BLOB_NULL_VALUE, NULL));
×
430
          }
431
        } else if (COL_VAL_IS_NONE(&colValArray[colValIndex])) {  // NONE
×
432
          if (IS_STR_DATA_BLOB(schema->columns[i].type)) {
×
433
            TAOS_CHECK_RETURN(addEmptyItemToBlobSet(pBlobSet, TSDB_DATA_BLOB_NULL_VALUE, NULL));
×
434
          }
435
          ROW_SET_BITMAP(bitmap, sinfo->tupleFlag, i - 1, BIT_FLG_NONE);
×
436
        }
437

438
        colValIndex++;
×
439
        break;
×
440
      } else if (colValArray[colValIndex].cid > schema->columns[i].colId) {  // NONE
×
441
        ROW_SET_BITMAP(bitmap, sinfo->tupleFlag, i - 1, BIT_FLG_NONE);
×
442
        break;
×
443
      } else {
444
        colValIndex++;
×
445
      }
446
    }
447
  }
448

449
  if (((*ppRow)->flag) == 0) {
×
450
    return TSDB_CODE_INVALID_PARA;
×
451
  }
452

453
  return 0;
×
454
}
455

456
static int32_t tRowBuildTupleWithBlob2(SArray *aColVal, const SRowBuildScanInfo *sinfo, const STSchema *schema,
×
457
                                       SRow **ppRow, SBlobSet *pSrcBlobSet, SBlobSet *pDstBlobSet) {
458
  int32_t  code = 0;
×
459
  SColVal *colValArray = (SColVal *)TARRAY_DATA(aColVal);
×
460
  *ppRow = (SRow *)taosMemoryCalloc(1, sinfo->tupleRowSize);
×
461
  if (*ppRow == NULL) {
×
462
    return terrno;
×
463
  }
464
  (*ppRow)->flag = sinfo->tupleFlag;
×
465
  (*ppRow)->numOfPKs = sinfo->numOfPKs;
×
466
  (*ppRow)->sver = schema->version;
×
467
  (*ppRow)->len = sinfo->tupleRowSize;
×
468
  (*ppRow)->ts = colValArray[0].value.val;
×
469

470
  if (((*ppRow)->flag) == 0) {
×
471
    return TSDB_CODE_INVALID_PARA;
×
472
  }
473
  if (sinfo->tupleFlag == HAS_NONE || sinfo->tupleFlag == HAS_NULL) {
×
474
    code = addEmptyItemToBlobSet(pDstBlobSet, TSDB_DATA_BLOB_NULL_VALUE, NULL);
×
475
    return code;
×
476
  }
477

478
  uint8_t *primaryKeys = (*ppRow)->data;
×
479
  uint8_t *bitmap = primaryKeys + sinfo->tuplePKSize;
×
480
  uint8_t *fixed = bitmap + sinfo->tupleBitmapSize;
×
481
  uint8_t *varlen = fixed + sinfo->tupleFixedSize;
×
482
  uint8_t *start = varlen;
×
483

484
  // primary keys
485
  for (int32_t i = 0; i < sinfo->numOfPKs; i++) {
×
486
    primaryKeys += tPutPrimaryKeyIndex(primaryKeys, sinfo->tupleIndices + i);
×
487
  }
488

489
  // bitmap + fixed + varlen
490
  int32_t numOfColVals = TARRAY_SIZE(aColVal);
×
491
  int32_t colValIndex = 1;
×
492
  for (int32_t i = 1; i < schema->numOfCols; i++) {
×
493
    for (;;) {
×
494
      if (colValIndex >= numOfColVals) {  // NONE
×
495
        ROW_SET_BITMAP(bitmap, sinfo->tupleFlag, i - 1, BIT_FLG_NONE);
×
496
        break;
×
497
      }
498

499
      int8_t hasBlob = 0;
×
500
      if (colValArray[colValIndex].cid == schema->columns[i].colId) {
×
501
        if (COL_VAL_IS_VALUE(&colValArray[colValIndex])) {  // value
×
502
          ROW_SET_BITMAP(bitmap, sinfo->tupleFlag, i - 1, BIT_FLG_VALUE);
×
503

504
          if (IS_VAR_DATA_TYPE(schema->columns[i].type)) {
×
505
            if (IS_STR_DATA_BLOB(schema->columns[i].type)) {
×
506
              hasBlob = 1;
×
507
            }
508
            *(int32_t *)(fixed + schema->columns[i].offset) = varlen - fixed - sinfo->tupleFixedSize;
×
509
            varlen += tPutU32v(varlen, colValArray[colValIndex].value.nData);
×
510
            if (colValArray[colValIndex].value.nData) {
×
511
              if (hasBlob == 0) {
×
512
                (void)memcpy(varlen, colValArray[colValIndex].value.pData, colValArray[colValIndex].value.nData);
×
513
                varlen += colValArray[colValIndex].value.nData;
×
514
              } else {
515
                uint64_t seq = 0;
×
516
                if (tGetU64(colValArray[colValIndex].value.pData, &seq) < 0) {
×
517
                  TAOS_CHECK_RETURN(TSDB_CODE_INVALID_PARA);
×
518
                }
519
                SBlobItem item = {0};
×
520

521
                code = tBlobSetGet(pSrcBlobSet, seq, &item);
×
522
                TAOS_CHECK_RETURN(code);
×
523

524
                code = tBlobSetPush(pDstBlobSet, &item, &seq, 0);
×
525
                TAOS_CHECK_RETURN(code);
×
526
                varlen += tPutU64(varlen, seq);
×
527
              }
528
            } else {
529
              if (hasBlob) TAOS_CHECK_RETURN(addEmptyItemToBlobSet(pDstBlobSet, TSDB_DATA_BLOB_EMPTY_VALUE, NULL));
×
530
            }
531
          } else {
532
            (void)memcpy(fixed + schema->columns[i].offset,
×
533
                         VALUE_GET_DATUM(&colValArray[colValIndex].value, schema->columns[i].type),
×
534
                         tDataTypes[schema->columns[i].type].bytes);
×
535
          }
536
        } else if (COL_VAL_IS_NULL(&colValArray[colValIndex])) {  // NULL
×
537
          ROW_SET_BITMAP(bitmap, sinfo->tupleFlag, i - 1, BIT_FLG_NULL);
×
538
          if (IS_STR_DATA_BLOB(schema->columns[i].type)) {
×
539
            TAOS_CHECK_RETURN(addEmptyItemToBlobSet(pDstBlobSet, TSDB_DATA_BLOB_NULL_VALUE, NULL));
×
540
          }
541
        } else if (COL_VAL_IS_NONE(&colValArray[colValIndex])) {  // NONE
×
542
          if (IS_STR_DATA_BLOB(schema->columns[i].type)) {
×
543
            TAOS_CHECK_RETURN(addEmptyItemToBlobSet(pDstBlobSet, TSDB_DATA_BLOB_NULL_VALUE, NULL));
×
544
          }
545
          ROW_SET_BITMAP(bitmap, sinfo->tupleFlag, i - 1, BIT_FLG_NONE);
×
546
        }
547

548
        colValIndex++;
×
549
        break;
×
550
      } else if (colValArray[colValIndex].cid > schema->columns[i].colId) {  // NONE
×
551
        ROW_SET_BITMAP(bitmap, sinfo->tupleFlag, i - 1, BIT_FLG_NONE);
×
552
        break;
×
553
      } else {
554
        colValIndex++;
×
555
      }
556
    }
557
  }
558

559
  if (((*ppRow)->flag) == 0) {
×
560
    return TSDB_CODE_INVALID_PARA;
×
561
  }
562

563
  return 0;
×
564
}
565
static FORCE_INLINE void tRowBuildKVRowSetIndex(uint8_t flag, SKVIdx *indices, uint32_t offset) {
566
  if (flag & KV_FLG_LIT) {
2,147,483,647✔
567
    ((uint8_t *)indices->idx)[indices->nCol] = (uint8_t)offset;
2,147,483,647✔
568
  } else if (flag & KV_FLG_MID) {
×
569
    ((uint16_t *)indices->idx)[indices->nCol] = (uint16_t)offset;
387,267✔
570
  } else {
571
    ((uint32_t *)indices->idx)[indices->nCol] = (uint32_t)offset;
×
572
  }
573
  indices->nCol++;
2,147,483,647✔
574
}
2,147,483,647✔
575

576
static int32_t tRowBuildKVRow(SArray *aColVal, const SRowBuildScanInfo *sinfo, const STSchema *schema, SRow **ppRow) {
607,739,970✔
577
  SColVal *colValArray = (SColVal *)TARRAY_DATA(aColVal);
607,739,970✔
578

579
  *ppRow = (SRow *)taosMemoryCalloc(1, sinfo->kvRowSize);
607,739,970!
580
  if (*ppRow == NULL) {
608,311,429!
581
    return terrno;
×
582
  }
583
  (*ppRow)->flag = sinfo->kvFlag;
608,311,429✔
584
  (*ppRow)->numOfPKs = sinfo->numOfPKs;
608,311,429✔
585
  (*ppRow)->sver = schema->version;
608,311,429✔
586
  (*ppRow)->len = sinfo->kvRowSize;
608,311,429✔
587
  (*ppRow)->ts = VALUE_GET_TRIVIAL_DATUM(&colValArray[0].value);
608,311,429✔
588

589
  if (!(sinfo->flag != HAS_NONE && sinfo->flag != HAS_NULL)) {
608,311,429!
590
    return TSDB_CODE_INVALID_PARA;
×
591
  }
592

593
  uint8_t *primaryKeys = (*ppRow)->data;
608,316,123✔
594
  SKVIdx  *indices = (SKVIdx *)(primaryKeys + sinfo->kvPKSize);
608,316,123✔
595
  uint8_t *payload = primaryKeys + sinfo->kvPKSize + sinfo->kvIndexSize;
608,316,123✔
596
  uint32_t payloadSize = 0;
608,316,123✔
597

598
  // primary keys
599
  for (int32_t i = 0; i < sinfo->numOfPKs; i++) {
608,648,334✔
600
    primaryKeys += tPutPrimaryKeyIndex(primaryKeys, sinfo->kvIndices + i);
332,216✔
601
  }
602

603
  int32_t numOfColVals = TARRAY_SIZE(aColVal);
608,316,118✔
604
  int32_t colValIndex = 1;
608,316,118✔
605
  for (int32_t i = 1; i < schema->numOfCols; i++) {
2,147,483,647✔
606
    for (;;) {
607
      if (colValIndex >= numOfColVals) {  // NONE
2,147,483,647✔
608
        break;
84✔
609
      }
610

611
      if (colValArray[colValIndex].cid == schema->columns[i].colId) {
2,147,483,647!
612
        if (COL_VAL_IS_VALUE(&colValArray[colValIndex])) {  // value
2,147,483,647✔
613
          tRowBuildKVRowSetIndex(sinfo->kvFlag, indices, payloadSize);
2,147,483,647!
614
          if (IS_VAR_DATA_TYPE(schema->columns[i].type)) {
2,147,483,647!
615
            payloadSize += tPutI16v(payload + payloadSize, colValArray[colValIndex].cid);
818,826,650✔
616
            payloadSize += tPutU32v(payload + payloadSize, colValArray[colValIndex].value.nData);
818,826,650✔
617
            if (colValArray[colValIndex].value.nData > 0) {
818,826,650!
618
              (void)memcpy(payload + payloadSize, colValArray[colValIndex].value.pData,
842,732,894✔
619
                           colValArray[colValIndex].value.nData);
842,732,894✔
620
              payloadSize += colValArray[colValIndex].value.nData;
842,732,894✔
621
            }
622
          } else {
623
            payloadSize += tPutI16v(payload + payloadSize, colValArray[colValIndex].cid);
2,147,483,647✔
624
            (void)memcpy(payload + payloadSize,
2,147,483,647✔
625
                         VALUE_GET_DATUM(&colValArray[colValIndex].value, schema->columns[i].type),
2,147,483,647!
626
                         tDataTypes[schema->columns[i].type].bytes);
2,147,483,647✔
627
            payloadSize += tDataTypes[schema->columns[i].type].bytes;
2,147,483,647✔
628
          }
629
        } else if (COL_VAL_IS_NULL(&colValArray[colValIndex])) {  // NULL
2,147,483,647✔
630
          tRowBuildKVRowSetIndex(sinfo->kvFlag, indices, payloadSize);
1,509,481✔
631
          payloadSize += tPutI16v(payload + payloadSize, -schema->columns[i].colId);
3,018,962✔
632
        }
633

634
        colValIndex++;
2,147,483,647✔
635
        break;
2,147,483,647✔
636
      } else if (colValArray[colValIndex].cid > schema->columns[i].colId) {  // NONE
×
637
        break;
228✔
638
      } else {
639
        colValIndex++;
×
640
      }
641
    }
642
  }
643
  return 0;
608,316,118✔
644
}
645

646
static int32_t tRowBuildKVRowWithBlob(SArray *aColVal, const SRowBuildScanInfo *sinfo, const STSchema *schema,
×
647
                                      SRow **ppRow, SBlobSet *ppBlobSet) {
648
  SColVal *colValArray = (SColVal *)TARRAY_DATA(aColVal);
×
649

650
  *ppRow = (SRow *)taosMemoryCalloc(1, sinfo->kvRowSize);
×
651
  if (*ppRow == NULL) {
×
652
    return terrno;
×
653
  }
654
  (*ppRow)->flag = sinfo->kvFlag;
×
655
  (*ppRow)->numOfPKs = sinfo->numOfPKs;
×
656
  (*ppRow)->sver = schema->version;
×
657
  (*ppRow)->len = sinfo->kvRowSize;
×
658
  (*ppRow)->ts = colValArray[0].value.val;
×
659

660
  if (!(sinfo->flag != HAS_NONE && sinfo->flag != HAS_NULL)) {
×
661
    return TSDB_CODE_INVALID_PARA;
×
662
  }
663

664
  if (((*ppRow)->flag) == 0) {
×
665
    return TSDB_CODE_INVALID_PARA;
×
666
  }
667

668
  uint8_t *primaryKeys = (*ppRow)->data;
×
669
  SKVIdx  *indices = (SKVIdx *)(primaryKeys + sinfo->kvPKSize);
×
670
  uint8_t *payload = primaryKeys + sinfo->kvPKSize + sinfo->kvIndexSize;
×
671
  uint32_t payloadSize = 0;
×
672

673
  // primary keys
674
  for (int32_t i = 0; i < sinfo->numOfPKs; i++) {
×
675
    primaryKeys += tPutPrimaryKeyIndex(primaryKeys, sinfo->kvIndices + i);
×
676
  }
677

678
  int32_t numOfColVals = TARRAY_SIZE(aColVal);
×
679
  int32_t colValIndex = 1;
×
680
  int8_t  firstBlobCol = 1;
×
681
  for (int32_t i = 1; i < schema->numOfCols; i++) {
×
682
    for (;;) {
×
683
      if (colValIndex >= numOfColVals) {  // NONE
×
684
        break;
×
685
      }
686
      uint8_t hasBlob = 0;
×
687

688
      if (colValArray[colValIndex].cid == schema->columns[i].colId) {
×
689
        if (COL_VAL_IS_VALUE(&colValArray[colValIndex])) {  // value
×
690
          tRowBuildKVRowSetIndex(sinfo->kvFlag, indices, payloadSize);
×
691
          if (IS_VAR_DATA_TYPE(schema->columns[i].type)) {
×
692
            if (IS_STR_DATA_BLOB(schema->columns[i].type)) {
×
693
              hasBlob = 1;
×
694
            }
695
            payloadSize += tPutI16v(payload + payloadSize, colValArray[colValIndex].cid);
×
696
            payloadSize += tPutU32v(payload + payloadSize, colValArray[colValIndex].value.nData);
×
697
            if (colValArray[colValIndex].value.nData > 0) {
×
698
              if (hasBlob == 0) {
×
699
                (void)memcpy(payload + payloadSize, colValArray[colValIndex].value.pData,
×
700
                             colValArray[colValIndex].value.nData);
×
701
                payloadSize += colValArray[colValIndex].value.nData;
×
702
              } else {
703
                uint64_t  seq = 0;
×
704
                uint32_t  seqOffset = payloadSize + payload - (*ppRow)->data;
×
705
                SBlobItem item = {.seqOffsetInRow = seqOffset,
×
706
                                  .data = colValArray[colValIndex].value.pData,
×
707
                                  .len = colValArray[colValIndex].value.nData,
×
708
                                  .type = TSDB_DATA_BLOB_VALUE};
709
                int32_t   code = tBlobSetPush(ppBlobSet, &item, &seq, 0);
×
710
                if (code != 0) return code;
×
711
                payloadSize += tPutU64(payload + payloadSize, seq);
×
712
              }
713
            } else {
714
              if (hasBlob) {
×
715
                TAOS_CHECK_RETURN(addEmptyItemToBlobSet(ppBlobSet, TSDB_DATA_BLOB_EMPTY_VALUE, NULL));
×
716
              }
717
            }
718
          } else {
719
            payloadSize += tPutI16v(payload + payloadSize, colValArray[colValIndex].cid);
×
720
            (void)memcpy(payload + payloadSize, &colValArray[colValIndex].value.val,
×
721
                         tDataTypes[schema->columns[i].type].bytes);
×
722
            payloadSize += tDataTypes[schema->columns[i].type].bytes;
×
723
          }
724
        } else if (COL_VAL_IS_NULL(&colValArray[colValIndex])) {  // NULL
×
725
          tRowBuildKVRowSetIndex(sinfo->kvFlag, indices, payloadSize);
×
726
          payloadSize += tPutI16v(payload + payloadSize, -schema->columns[i].colId);
×
727
          if (IS_STR_DATA_BLOB(schema->columns[i].type)) {
×
728
            TAOS_CHECK_RETURN(addEmptyItemToBlobSet(ppBlobSet, TSDB_DATA_BLOB_NULL_VALUE, NULL));
×
729
          }
730
        }
731
        colValIndex++;
×
732
        break;
×
733
      } else if (colValArray[colValIndex].cid > schema->columns[i].colId) {  // NONE
×
734
        break;
×
735
      } else {
736
        colValIndex++;
×
737
      }
738
    }
739
  }
740

741
  if (((*ppRow)->flag) == 0) {
×
742
    return TSDB_CODE_INVALID_PARA;
×
743
  }
744
  return 0;
×
745
}
746

747
static int32_t tRowBuildKVRowWithBlob2(SArray *aColVal, const SRowBuildScanInfo *sinfo, const STSchema *schema,
×
748
                                       SRow **ppRow, SBlobSet *pSrcBlobSet, SBlobSet *pDstBlobSet) {
749
  SColVal *colValArray = (SColVal *)TARRAY_DATA(aColVal);
×
750

751
  *ppRow = (SRow *)taosMemoryCalloc(1, sinfo->kvRowSize);
×
752
  if (*ppRow == NULL) {
×
753
    return terrno;
×
754
  }
755
  (*ppRow)->flag = sinfo->kvFlag;
×
756
  (*ppRow)->numOfPKs = sinfo->numOfPKs;
×
757
  (*ppRow)->sver = schema->version;
×
758
  (*ppRow)->len = sinfo->kvRowSize;
×
759
  (*ppRow)->ts = colValArray[0].value.val;
×
760

761
  if (!(sinfo->flag != HAS_NONE && sinfo->flag != HAS_NULL)) {
×
762
    return TSDB_CODE_INVALID_PARA;
×
763
  }
764

765
  if (((*ppRow)->flag) == 0) {
×
766
    return TSDB_CODE_INVALID_PARA;
×
767
  }
768

769
  uint8_t *primaryKeys = (*ppRow)->data;
×
770
  SKVIdx  *indices = (SKVIdx *)(primaryKeys + sinfo->kvPKSize);
×
771
  uint8_t *payload = primaryKeys + sinfo->kvPKSize + sinfo->kvIndexSize;
×
772
  uint32_t payloadSize = 0;
×
773

774
  // primary keys
775
  for (int32_t i = 0; i < sinfo->numOfPKs; i++) {
×
776
    primaryKeys += tPutPrimaryKeyIndex(primaryKeys, sinfo->kvIndices + i);
×
777
  }
778

779
  int32_t numOfColVals = TARRAY_SIZE(aColVal);
×
780
  int32_t colValIndex = 1;
×
781
  for (int32_t i = 1; i < schema->numOfCols; i++) {
×
782
    for (;;) {
×
783
      if (colValIndex >= numOfColVals) {  // NONE
×
784
        break;
×
785
      }
786
      uint8_t hasBlob = 0;
×
787

788
      if (colValArray[colValIndex].cid == schema->columns[i].colId) {
×
789
        if (COL_VAL_IS_VALUE(&colValArray[colValIndex])) {  // value
×
790
          tRowBuildKVRowSetIndex(sinfo->kvFlag, indices, payloadSize);
×
791
          if (IS_VAR_DATA_TYPE(schema->columns[i].type)) {
×
792
            if (IS_STR_DATA_BLOB(schema->columns[i].type)) {
×
793
              hasBlob = 1;
×
794
            }
795
            payloadSize += tPutI16v(payload + payloadSize, colValArray[colValIndex].cid);
×
796
            payloadSize += tPutU32v(payload + payloadSize, colValArray[colValIndex].value.nData);
×
797
            if (colValArray[colValIndex].value.nData > 0) {
×
798
              if (hasBlob == 0) {
×
799
                (void)memcpy(payload + payloadSize, colValArray[colValIndex].value.pData,
×
800
                             colValArray[colValIndex].value.nData);
×
801
                payloadSize += colValArray[colValIndex].value.nData;
×
802
              } else {
803
                uint64_t seq = 0;
×
804
                if (tGetU64(colValArray[colValIndex].value.pData, &seq) < 0) {
×
805
                  TAOS_CHECK_RETURN(TSDB_CODE_INVALID_PARA);
×
806
                }
807
                SBlobItem item = {0};
×
808

809
                int32_t code = tBlobSetGet(pSrcBlobSet, seq, &item);
×
810
                TAOS_CHECK_RETURN(code);
×
811

812
                code = tBlobSetPush(pDstBlobSet, &item, &seq, 0);
×
813
                TAOS_CHECK_RETURN(code);
×
814

815
                payloadSize += tPutU64(payload + payloadSize, seq);
×
816
              }
817
            } else {
818
              if (hasBlob) {
×
819
                TAOS_CHECK_RETURN(addEmptyItemToBlobSet(pDstBlobSet, TSDB_DATA_BLOB_EMPTY_VALUE, NULL));
×
820
              }
821
            }
822
          } else {
823
            payloadSize += tPutI16v(payload + payloadSize, colValArray[colValIndex].cid);
×
824
            (void)memcpy(payload + payloadSize, &colValArray[colValIndex].value.val,
×
825
                         tDataTypes[schema->columns[i].type].bytes);
×
826
            payloadSize += tDataTypes[schema->columns[i].type].bytes;
×
827
          }
828
        } else if (COL_VAL_IS_NULL(&colValArray[colValIndex])) {  // NULL
×
829
          tRowBuildKVRowSetIndex(sinfo->kvFlag, indices, payloadSize);
×
830
          payloadSize += tPutI16v(payload + payloadSize, -schema->columns[i].colId);
×
831
          if (IS_STR_DATA_BLOB(schema->columns[i].type)) {
×
832
            TAOS_CHECK_RETURN(addEmptyItemToBlobSet(pDstBlobSet, TSDB_DATA_BLOB_NULL_VALUE, NULL));
×
833
          }
834
        }
835
        colValIndex++;
×
836
        break;
×
837
      } else if (colValArray[colValIndex].cid > schema->columns[i].colId) {  // NONE
×
838
        break;
×
839
      } else {
840
        colValIndex++;
×
841
      }
842
    }
843
  }
844

845
  if (((*ppRow)->flag) == 0) {
×
846
    return TSDB_CODE_INVALID_PARA;
×
847
  }
848
  return 0;
×
849
}
850
int32_t tRowBuild(SArray *aColVal, const STSchema *pTSchema, SRow **ppRow, SRowBuildScanInfo *sinfo) {
2,129,072,973✔
851
  int32_t code;
852
  code = tRowBuildScan(aColVal, pTSchema, sinfo);
2,129,072,973✔
853
  if (code) return code;
2,136,046,154✔
854

855
  if (sinfo->tupleRowSize <= sinfo->kvRowSize) {
2,136,046,142✔
856
    code = tRowBuildTupleRow(aColVal, sinfo, pTSchema, ppRow);
1,528,996,311✔
857
  } else {
858
    code = tRowBuildKVRow(aColVal, sinfo, pTSchema, ppRow);
607,049,831✔
859
  }
860
  return code;
2,137,767,182✔
861
}
862

863
int32_t tBlobRowCreate(int64_t cap, int8_t type, SBlobSet **ppBlobRow) {
×
864
  int32_t    lino = 0;
×
865
  int32_t    code = 0;
×
866
  SBlobSet *p = taosMemCalloc(1, sizeof(SBlobSet));
×
867
  if (p == NULL) {
×
868
    return terrno;
×
869
  }
870

871
  p->type = type;
×
872
  p->pSeqTable = taosArrayInit(128, sizeof(SBlobValue));
×
873
  if (p->pSeqTable == NULL) {
×
874
    TAOS_CHECK_EXIT(terrno);
×
875
  }
876

877
  p->data = taosMemCalloc(1024, cap * sizeof(uint8_t));
×
878
  if (p->data == NULL) {
×
879
    TAOS_CHECK_EXIT(terrno);
×
880
  }
881

882
  p->pSeqToffset = taosHashInit(128, taosGetDefaultHashFunction(TSDB_DATA_TYPE_UBIGINT), false, HASH_NO_LOCK);
×
883
  if (p->pSeqToffset == NULL) {
×
884
    TAOS_CHECK_EXIT(terrno);
×
885
  }
886

887
  p->pSet = taosArrayInit(4, sizeof(int32_t));
×
888
  if (p->pSet == NULL) {
×
889
    TAOS_CHECK_EXIT(terrno);
×
890
  }
891

892
  p->cap = cap;
×
893
  p->len = 0;
×
894
  p->seq = 1;
×
895

896
  *ppBlobRow = p;
×
897
_exit:
×
898
  if (code != 0) {
×
899
    taosHashCleanup(p->pSeqToffset);
×
900
    taosArrayDestroy(p->pSeqTable);
×
901
    taosArrayDestroy(p->pSet);
×
902

903
    taosMemoryFree(p->data);
×
904
    taosMemoryFree(p);
×
905
  }
906
  uDebug("create blob row %p", p);
×
907
  return code;
×
908
}
909
int32_t tBlobRowPush(SBlobSet *pBlobRow, SBlobItem *pItem, uint64_t *seq, int8_t nextRow) {
×
910
  if (pBlobRow == NULL || pItem == NULL || seq == NULL) {
×
911
    return TSDB_CODE_INVALID_PARA;
×
912
  }
913
  uTrace("push blob %p, seqOffsetInRow %d, dataLen %d, nextRow %d", pBlobRow, pItem->seqOffsetInRow, pItem->len,
×
914
         nextRow);
915
  int32_t  lino = 0;
×
916
  int32_t  code = 0;
×
917
  uint64_t offset;
918
  uint32_t len = pItem->len;
×
919
  uint8_t *data = pItem->data;
×
920
  int32_t  dataOffset = pItem->seqOffsetInRow;
×
921

922
  uint64_t tlen = pBlobRow->len + len;
×
923
  if (tlen > pBlobRow->cap) {
×
924
    int64_t cap = pBlobRow->cap;
×
925
    // opt later
926
    do {
927
      cap = cap * 2;
×
928
    } while (tlen > cap);
×
929

930
    uint8_t *data = taosMemRealloc(pBlobRow->data, cap);
×
931
    if (data == NULL) {
×
932
      return terrno;
×
933
    }
934
    pBlobRow->data = data;
×
935
    pBlobRow->cap = cap;
×
936
  }
937
  if (len > 0) {
×
938
    (void)memcpy(pBlobRow->data + pBlobRow->len, data, len);
×
939
  }
940

941
  offset = pBlobRow->len;
×
942
  pBlobRow->len += len;
×
943

944
  pBlobRow->seq++;
×
945
  *seq = pBlobRow->seq;
×
946

947
  SBlobValue value = {.offset = offset, .len = len, .dataOffset = dataOffset, .nextRow = nextRow};
×
948
  if (taosArrayPush(pBlobRow->pSeqTable, &value) == NULL) {
×
949
    TAOS_CHECK_EXIT(terrno);
×
950
  }
951

952
  int32_t sz = taosArrayGetSize(pBlobRow->pSeqTable);
×
953
  code = taosHashPut(pBlobRow->pSeqToffset, seq, sizeof(int64_t), &sz, sizeof(int32_t));
×
954
  if (code != 0) {
×
955
    TAOS_CHECK_EXIT(code);
×
956
  }
957

958
_exit:
×
959
  return code;
×
960
}
961
int32_t tBlobRowUpdate(SBlobSet *pBlobRow, uint64_t seq, SBlobItem *pItem) {
×
962
  int32_t code = 0;
×
963
  return code;
×
964
  if (pBlobRow == NULL || pItem == NULL) {
965
    return TSDB_CODE_INVALID_PARA;
966
  }
967
  if (pBlobRow->pSeqToffset == NULL || pBlobRow->pSeqTable == NULL) {
968
    return TSDB_CODE_INVALID_PARA;
969
  }
970

971
  return code;
972
}
973
int32_t tBlobRowGet(SBlobSet *pBlobRow, uint64_t seq, SBlobItem *pItem) {
×
974
  if (pBlobRow == NULL || pItem == NULL) {
×
975
    return TSDB_CODE_INVALID_PARA;
×
976
  }
977
  if (pBlobRow->pSeqToffset == NULL || pBlobRow->pSeqTable == NULL) {
×
978
    return TSDB_CODE_INVALID_PARA;
×
979
  }
980

981
  int32_t code = 0;
×
982
  int32_t len = 0;
×
983
  int32_t dataOffset = 0;
×
984
  int8_t  nextRow = 0;
×
985

986
  int32_t *offset = (int32_t *)taosHashGet(pBlobRow->pSeqToffset, &seq, sizeof(int64_t));
×
987
  if (offset == NULL) {
×
988
    return TSDB_CODE_INVALID_PARA;
×
989
  }
990

991
  SBlobValue *value = taosArrayGet(pBlobRow->pSeqTable, *offset - 1);
×
992
  if (value == NULL) {
×
993
    return TSDB_CODE_INVALID_PARA;
×
994
  }
995

996
  len = value->len;
×
997
  dataOffset = value->dataOffset;
×
998
  nextRow = value->nextRow;
×
999

1000
  pItem->seqOffsetInRow = dataOffset + pBlobRow->data - (uint8_t *)pBlobRow;
×
1001
  pItem->len = len;
×
1002
  pItem->data = pBlobRow->data + value->offset;
×
1003

1004
  return code;
×
1005
}
1006

1007
int32_t tBlobRowSize(SBlobSet *pBlobRow) {
×
1008
  if (pBlobRow == NULL) return 0;
×
1009
  return taosArrayGetSize(pBlobRow->pSeqTable);
×
1010
}
1011
int32_t tBlobRowEnd(SBlobSet *pBlobRow) {
×
1012
  if (pBlobRow == NULL) return 0;
×
1013
  int32_t sz = taosArrayGetSize(pBlobRow->pSeqTable);
×
1014
  if (taosArrayPush(pBlobRow->pSet, &sz) == NULL) {
×
1015
    return terrno;
×
1016
  }
1017
  return 0;
×
1018
}
1019
int32_t tBlobRowRebuild(SBlobSet *p, int32_t sRow, int32_t nrow, SBlobSet **pDst) {
×
1020
  int32_t code = 0;
×
1021

1022
  code = tBlobRowCreate(p->cap, p->type, pDst);
×
1023
  if (code != 0) {
×
1024
    return code;
×
1025
  }
1026

1027
  SBlobSet *pBlobRow = *pDst;
×
1028
  if (p->pSeqToffset == NULL || p->pSeqTable == NULL || p->data == NULL) {
×
1029
    return TSDB_CODE_INVALID_PARA;
×
1030
  }
1031

1032
  uint64_t seq = 0;
×
1033
  int32_t  count = 0;
×
1034
  for (int32_t i = sRow; i < taosArrayGetSize(p->pSeqTable); i++) {
×
1035
    SBlobValue *pValue = taosArrayGet(p->pSeqTable, i);
×
1036
    uint8_t    *data = p->data + pValue->offset;
×
1037
    int32_t     len = pValue->len;
×
1038

1039
    SBlobItem item = {.data = data, .len = len, .seqOffsetInRow = pValue->dataOffset};
×
1040

1041
    code = tBlobRowPush(pBlobRow, &item, &seq, pValue->nextRow);
×
1042
    count++;
×
1043
    if (count >= nrow) {
×
1044
      break;
×
1045
    }
1046
  }
1047

1048
  return code;
×
1049
}
1050

1051
int32_t tBlobRowDestroy(SBlobSet *pBlobRow) {
×
1052
  if (pBlobRow == NULL) return 0;
×
1053
  int32_t code = 0;
×
1054
  uTrace("destroy blob row, seqTable size %p", pBlobRow);
×
1055
  taosMemoryFree(pBlobRow->data);
×
1056
  taosArrayDestroy(pBlobRow->pSeqTable);
×
1057
  taosHashCleanup(pBlobRow->pSeqToffset);
×
1058
  taosArrayDestroy(pBlobRow->pSet);
×
1059
  taosMemoryFree(pBlobRow);
×
1060
  return code;
×
1061
}
1062
int32_t tBlobRowClear(SBlobSet *pBlobRow) {
×
1063
  if (pBlobRow == NULL) return 0;
×
1064
  int32_t code = 0;
×
1065
  uTrace("clear blob row, seqTable size %p", pBlobRow);
×
1066
  taosArrayClear(pBlobRow->pSeqTable);
×
1067
  taosHashClear(pBlobRow->pSeqToffset);
×
1068
  pBlobRow->len = 0;
×
1069
  pBlobRow->seq = 1;
×
1070
  return code;
×
1071
}
1072

1073
int32_t tRowBuildWithBlob(SArray *aColVal, const STSchema *pTSchema, SRow **ppRow, SBlobSet *pBlobSet,
×
1074
                          SRowBuildScanInfo *sinfo) {
1075
  int32_t code;
1076

1077
  code = tRowBuildScan(aColVal, pTSchema, sinfo);
×
1078
  if (code) return code;
×
1079

1080
  if (sinfo->tupleRowSize <= sinfo->kvRowSize) {
×
1081
    code = tRowBuildTupleWithBlob(aColVal, sinfo, pTSchema, ppRow, pBlobSet);
×
1082
  } else {
1083
    code = tRowBuildKVRowWithBlob(aColVal, sinfo, pTSchema, ppRow, pBlobSet);
×
1084
  }
1085

1086
  return code;
×
1087
}
1088

1089
int32_t tRowBuildWithBlob2(SArray *aColVal, const STSchema *pTSchema, SRow **ppRow, SBlobSet *pSrcBlobSet,
×
1090
                           SBlobSet *pDstBlobSet, SRowBuildScanInfo *sinfo) {
1091
  int32_t code;
1092

1093
  code = tRowBuildScan(aColVal, pTSchema, sinfo);
×
1094
  if (code) return code;
×
1095

1096
  if (sinfo->tupleRowSize <= sinfo->kvRowSize) {
×
1097
    code = tRowBuildTupleWithBlob2(aColVal, sinfo, pTSchema, ppRow, pSrcBlobSet, pDstBlobSet);
×
1098
  } else {
1099
    code = tRowBuildKVRowWithBlob2(aColVal, sinfo, pTSchema, ppRow, pSrcBlobSet, pDstBlobSet);
×
1100
  }
1101

1102
  return code;
×
1103
}
1104

1105
int32_t tBlobSetCreate(int64_t cap, int8_t type, SBlobSet **ppBlobSet) {
×
1106
  int32_t    lino = 0;
×
1107
  int32_t    code = 0;
×
1108
  SBlobSet  *p = taosMemCalloc(1, sizeof(SBlobSet));
×
1109
  if (p == NULL) {
×
1110
    return terrno;
×
1111
  }
1112

1113
  p->type = type;
×
1114
  p->pSeqTable = taosArrayInit(128, sizeof(SBlobValue));
×
1115
  if (p->pSeqTable == NULL) {
×
1116
    TAOS_CHECK_EXIT(terrno);
×
1117
  }
1118

1119
  p->data = taosMemCalloc(1024, cap * sizeof(uint8_t));
×
1120
  if (p->data == NULL) {
×
1121
    TAOS_CHECK_EXIT(terrno);
×
1122
  }
1123

1124
  p->pSeqToffset = taosHashInit(128, taosGetDefaultHashFunction(TSDB_DATA_TYPE_UBIGINT), false, HASH_NO_LOCK);
×
1125
  if (p->pSeqToffset == NULL) {
×
1126
    TAOS_CHECK_EXIT(terrno);
×
1127
  }
1128

1129
  p->pSet = taosArrayInit(4, sizeof(int32_t));
×
1130
  if (p->pSet == NULL) {
×
1131
    TAOS_CHECK_EXIT(terrno);
×
1132
  }
1133

1134
  p->cap = cap;
×
1135
  p->len = 0;
×
1136
  p->seq = 1;
×
1137

1138
  *ppBlobSet = p;
×
1139
_exit:
×
1140
  if (code != 0) {
×
1141
    taosHashCleanup(p->pSeqToffset);
×
1142
    taosArrayDestroy(p->pSeqTable);
×
1143
    taosArrayDestroy(p->pSet);
×
1144

1145
    taosMemoryFree(p->data);
×
1146
    taosMemoryFree(p);
×
1147
  }
1148
  uDebug("create blob row %p", p);
×
1149
  return code;
×
1150
}
1151
int32_t tBlobSetPush(SBlobSet *pBlobSet, SBlobItem *pItem, uint64_t *seq, int8_t nextRow) {
×
1152
  if (pBlobSet == NULL || pItem == NULL || seq == NULL) {
×
1153
    return TSDB_CODE_INVALID_PARA;
×
1154
  }
1155
  uTrace("push blob %p, seqOffsetInRow %d, dataLen %d, nextRow %d", pBlobSet, pItem->seqOffsetInRow, pItem->len,
×
1156
         nextRow);
1157
  int32_t  lino = 0;
×
1158
  int32_t  code = 0;
×
1159
  uint64_t offset;
1160
  uint32_t len = pItem->len;
×
1161
  uint8_t *data = pItem->data;
×
1162
  int32_t  dataOffset = pItem->seqOffsetInRow;
×
1163

1164
  uint64_t tlen = pBlobSet->len + len;
×
1165
  if (tlen > pBlobSet->cap) {
×
1166
    int64_t cap = pBlobSet->cap;
×
1167
    // opt later
1168
    do {
1169
      cap = cap * 2;
×
1170
    } while (tlen > cap);
×
1171

1172
    uint8_t *data = taosMemRealloc(pBlobSet->data, cap);
×
1173
    if (data == NULL) {
×
1174
      return terrno;
×
1175
    }
1176
    pBlobSet->data = data;
×
1177
    pBlobSet->cap = cap;
×
1178
  }
1179
  if (len > 0) {
×
1180
    (void)memcpy(pBlobSet->data + pBlobSet->len, data, len);
×
1181
  }
1182

1183
  offset = pBlobSet->len;
×
1184
  pBlobSet->len += len;
×
1185

1186
  pBlobSet->seq++;
×
1187
  *seq = pBlobSet->seq;
×
1188

1189
  SBlobValue value = {.offset = offset, .len = len, .dataOffset = dataOffset, .nextRow = nextRow, .type = pItem->type};
×
1190
  if (taosArrayPush(pBlobSet->pSeqTable, &value) == NULL) {
×
1191
    TAOS_CHECK_EXIT(terrno);
×
1192
  }
1193

1194
  int32_t sz = taosArrayGetSize(pBlobSet->pSeqTable);
×
1195
  code = taosHashPut(pBlobSet->pSeqToffset, seq, sizeof(int64_t), &sz, sizeof(int32_t));
×
1196
  if (code != 0) {
×
1197
    TAOS_CHECK_EXIT(code);
×
1198
  }
1199

1200
_exit:
×
1201
  return code;
×
1202
}
1203

1204
void tBlobSetSwap(SBlobSet *p1, SBlobSet *p2) {
×
1205
  SBlobSet t = {0};
×
1206

1207
  memcpy(&t, p1, sizeof(SBlobSet));
×
1208

1209
  memcpy(p1, p2, sizeof(SBlobSet));
×
1210
  memcpy(p2, &t, sizeof(SBlobSet));
×
1211
}
×
1212

1213
int32_t tBlobSetUpdate(SBlobSet *pBlobSet, uint64_t seq, SBlobItem *pItem) {
×
1214
  int32_t code = 0;
×
1215
  return code;
×
1216
  if (pBlobSet == NULL || pItem == NULL) {
1217
    return TSDB_CODE_INVALID_PARA;
1218
  }
1219
  if (pBlobSet->pSeqToffset == NULL || pBlobSet->pSeqTable == NULL) {
1220
    return TSDB_CODE_INVALID_PARA;
1221
  }
1222

1223
  return code;
1224
}
1225
int32_t tBlobSetGet(SBlobSet *pBlobSet, uint64_t seq, SBlobItem *pItem) {
×
1226
  if (pBlobSet == NULL || pItem == NULL) {
×
1227
    return TSDB_CODE_INVALID_PARA;
×
1228
  }
1229
  if (pBlobSet->pSeqToffset == NULL || pBlobSet->pSeqTable == NULL) {
×
1230
    return TSDB_CODE_INVALID_PARA;
×
1231
  }
1232

1233
  int32_t code = 0;
×
1234
  int32_t len = 0;
×
1235
  int32_t dataOffset = 0;
×
1236
  int8_t  nextRow = 0;
×
1237

1238
  int32_t *offset = (int32_t *)taosHashGet(pBlobSet->pSeqToffset, &seq, sizeof(int64_t));
×
1239
  if (offset == NULL) {
×
1240
    return TSDB_CODE_INVALID_PARA;
×
1241
  }
1242

1243
  SBlobValue *value = taosArrayGet(pBlobSet->pSeqTable, *offset - 1);
×
1244
  if (value == NULL) {
×
1245
    return TSDB_CODE_INVALID_PARA;
×
1246
  }
1247

1248
  len = value->len;
×
1249
  dataOffset = value->dataOffset;
×
1250
  nextRow = value->nextRow;
×
1251

1252
  pItem->seqOffsetInRow = value->dataOffset;
×
1253
  pItem->len = len;
×
1254
  pItem->data = pBlobSet->data + value->offset;
×
1255
  pItem->type = value->type;
×
1256

1257
  return code;
×
1258
}
1259

1260
int32_t tBlobSetSize(SBlobSet *pBlobSet) {
31,816,385✔
1261
  if (pBlobSet == NULL) return 0;
31,816,385!
1262
  return taosArrayGetSize(pBlobSet->pSeqTable);
×
1263
}
1264

1265
void tBlobSetDestroy(SBlobSet *pBlobSet) {
22,219✔
1266
  if (pBlobSet == NULL) return;
22,219!
1267
  uTrace("destroy blob row, seqTable size %p", pBlobSet);
×
1268
  taosMemoryFree(pBlobSet->data);
×
1269
  taosArrayDestroy(pBlobSet->pSeqTable);
×
1270
  taosHashCleanup(pBlobSet->pSeqToffset);
×
1271
  taosArrayDestroy(pBlobSet->pSet);
×
1272
  taosMemoryFree(pBlobSet);
×
1273
}
1274
int32_t tBlobSetClear(SBlobSet *pBlobSet) {
×
1275
  if (pBlobSet == NULL) return 0;
×
1276
  int32_t code = 0;
×
1277
  uTrace("clear blob row, seqTable size %p", pBlobSet);
×
1278
  taosArrayClear(pBlobSet->pSeqTable);
×
1279
  taosHashClear(pBlobSet->pSeqToffset);
×
1280
  pBlobSet->len = 0;
×
1281
  pBlobSet->seq = 1;
×
1282
  return code;
×
1283
}
1284

1285
static int32_t tBindInfoCompare(const void *p1, const void *p2, const void *param) {
37✔
1286
  if (((SBindInfo *)p1)->columnId < ((SBindInfo *)p2)->columnId) {
37✔
1287
    return -1;
7✔
1288
  } else if (((SBindInfo *)p1)->columnId > ((SBindInfo *)p2)->columnId) {
30!
1289
    return 1;
30✔
1290
  }
1291
  return 0;
×
1292
}
1293

1294
/* build rows to `rowArray` from bind
1295
 * `infos` is the bind information array
1296
 * `numOfInfos` is the number of bind information
1297
 * `infoSorted` is whether the bind information is sorted by column id
1298
 * `pTSchema` is the schema of the table
1299
 * `rowArray` is the array to store the rows
1300
 * `pOrdered` is the pointer to store ordered
1301
 * `pDupTs` is the pointer to store duplicateTs
1302
 */
1303
int32_t tRowBuildFromBind(SBindInfo *infos, int32_t numOfInfos, bool infoSorted, const STSchema *pTSchema,
5,624✔
1304
                          SArray *rowArray, bool *pOrdered, bool *pDupTs) {
1305
  if (infos == NULL || numOfInfos <= 0 || numOfInfos > pTSchema->numOfCols || pTSchema == NULL || rowArray == NULL) {
5,624!
1306
    return TSDB_CODE_INVALID_PARA;
×
1307
  }
1308

1309
  if (!infoSorted) {
5,631!
1310
    taosqsort_r(infos, numOfInfos, sizeof(SBindInfo), NULL, tBindInfoCompare);
×
1311
  }
1312

1313
  int32_t code = 0;
5,631✔
1314
  int32_t numOfRows = infos[0].bind->num;
5,631✔
1315
  SArray *colValArray;
1316
  SColVal colVal;
1317

1318
  if ((colValArray = taosArrayInit(numOfInfos, sizeof(SColVal))) == NULL) {
5,631!
1319
    return terrno;
×
1320
  }
1321

1322
  SRowKey rowKey, lastRowKey;
1323
  for (int32_t iRow = 0; iRow < numOfRows; iRow++) {
66,586✔
1324
    taosArrayClear(colValArray);
62,258✔
1325

1326
    for (int32_t iInfo = 0; iInfo < numOfInfos; iInfo++) {
736,703✔
1327
      if (infos[iInfo].bind->is_null && infos[iInfo].bind->is_null[iRow]) {
685,416✔
1328
        colVal = COL_VAL_NULL(infos[iInfo].columnId, infos[iInfo].type);
32✔
1329
      } else {
1330
        SValue value = {
685,384✔
1331
            .type = infos[iInfo].type,
685,384✔
1332
        };
1333
        if (IS_VAR_DATA_TYPE(infos[iInfo].type)) {
685,384!
1334
          value.nData = infos[iInfo].bind->length[iRow];
200,023✔
1335
          if (value.nData > pTSchema->columns[iInfo].bytes - VARSTR_HEADER_SIZE) {
200,023!
1336
            code = TSDB_CODE_INVALID_PARA;
×
1337
            goto _exit;
×
1338
          }
1339
          value.pData = (uint8_t *)infos[iInfo].bind->buffer + infos[iInfo].bind->buffer_length * iRow;
200,023✔
1340
        } else {
1341
          valueSetDatum(&value, infos[iInfo].type,
485,361✔
1342
                        (uint8_t *)infos[iInfo].bind->buffer + infos[iInfo].bind->buffer_length * iRow,
485,361✔
1343
                        infos[iInfo].bind->buffer_length);
485,361✔
1344
        }
1345
        colVal = COL_VAL_VALUE(infos[iInfo].columnId, value);
691,966✔
1346
      }
1347
      if (taosArrayPush(colValArray, &colVal) == NULL) {
674,471!
1348
        code = terrno;
×
1349
        goto _exit;
×
1350
      }
1351
    }
1352

1353
    SRow             *row;
1354
    SRowBuildScanInfo sinfo = {0};
51,287✔
1355
    if ((code = tRowBuild(colValArray, pTSchema, &row, &sinfo))) {
51,287!
1356
      goto _exit;
×
1357
    }
1358

1359
    if ((taosArrayPush(rowArray, &row)) == NULL) {
60,902!
1360
      code = terrno;
×
1361
      goto _exit;
×
1362
    }
1363

1364
    if (pOrdered && pDupTs) {
60,902!
1365
      tRowGetKey(row, &rowKey);
122,120!
1366
      if (iRow == 0) {
61,117✔
1367
        *pOrdered = true;
5,630✔
1368
        *pDupTs = false;
5,630✔
1369
      } else {
1370
        if (*pOrdered) {
55,487!
1371
          int32_t res = tRowKeyCompare(&rowKey, &lastRowKey);
55,558✔
1372
          *pOrdered = (res >= 0);
55,558✔
1373
          if (!*pDupTs) {
55,558!
1374
            *pDupTs = (res == 0);
55,559✔
1375
          }
1376
        }
1377
      }
1378
      lastRowKey = rowKey;
61,117✔
1379
    }
1380
  }
1381

1382
_exit:
4,328✔
1383
  taosArrayDestroy(colValArray);
4,328✔
1384
  return code;
5,627✔
1385
}
1386

1387
int32_t tRowGet(SRow *pRow, STSchema *pTSchema, int32_t iCol, SColVal *pColVal) {
2,147,483,647✔
1388
  if (!(iCol < pTSchema->numOfCols)) return TSDB_CODE_INVALID_PARA;
2,147,483,647!
1389
  if (!(pRow->sver == pTSchema->version)) return TSDB_CODE_INVALID_PARA;
2,147,483,647!
1390

1391
  STColumn *pTColumn = pTSchema->columns + iCol;
2,147,483,647✔
1392

1393
  if (iCol == 0) {
2,147,483,647✔
1394
    pColVal->cid = pTColumn->colId;
10,531,292✔
1395
    pColVal->value.type = pTColumn->type;
10,531,292✔
1396
    pColVal->flag = CV_FLAG_VALUE;
10,531,292✔
1397
    VALUE_SET_TRIVIAL_DATUM(&pColVal->value, pRow->ts);
10,531,292✔
1398
    return 0;
10,531,292✔
1399
  }
1400

1401
  if (pRow->flag == HAS_NONE) {
2,147,483,647✔
1402
    *pColVal = COL_VAL_NONE(pTColumn->colId, pTColumn->type);
2,541,882✔
1403
    return 0;
2,541,882✔
1404
  }
1405

1406
  if (pRow->flag == HAS_NULL) {
2,147,483,647✔
1407
    *pColVal = COL_VAL_NULL(pTColumn->colId, pTColumn->type);
1,038,435✔
1408
    return 0;
1,038,435✔
1409
  }
1410

1411
  SPrimaryKeyIndex index;
1412
  uint8_t         *data = pRow->data;
2,147,483,647✔
1413
  for (int32_t i = 0; i < pRow->numOfPKs; i++) {
2,147,483,647✔
1414
    data += tGetPrimaryKeyIndex(data, &index);
39,192,031✔
1415
  }
1416

1417
  if (pRow->flag >> 4) {  // KV Row
2,147,483,647✔
1418
    SKVIdx  *pIdx = (SKVIdx *)data;
2,147,483,647✔
1419
    uint8_t *pv = NULL;
2,147,483,647✔
1420

1421
    if (pRow->flag & KV_FLG_LIT) {
2,147,483,647✔
1422
      pv = pIdx->idx + pIdx->nCol;
2,147,483,647✔
1423
    } else if (pRow->flag & KV_FLG_MID) {
35,211,903!
1424
      pv = pIdx->idx + (pIdx->nCol << 1);
99,777,251✔
1425
    } else {
1426
      pv = pIdx->idx + (pIdx->nCol << 2);
×
1427
    }
1428

1429
    int16_t lidx = 0;
2,147,483,647✔
1430
    int16_t ridx = pIdx->nCol - 1;
2,147,483,647✔
1431
    while (lidx <= ridx) {
2,147,483,647✔
1432
      int16_t  mid = (lidx + ridx) >> 1;
2,147,483,647✔
1433
      uint8_t *pData = NULL;
2,147,483,647✔
1434
      if (pRow->flag & KV_FLG_LIT) {
2,147,483,647✔
1435
        pData = pv + ((uint8_t *)pIdx->idx)[mid];
2,147,483,647✔
1436
      } else if (pRow->flag & KV_FLG_MID) {
423,280,331!
1437
        pData = pv + ((uint16_t *)pIdx->idx)[mid];
423,465,404✔
1438
      } else {
1439
        pData = pv + ((uint32_t *)pIdx->idx)[mid];
×
1440
      }
1441

1442
      int16_t cid;
1443
      pData += tGetI16v(pData, &cid);
2,147,483,647✔
1444

1445
      if (TABS(cid) == pTColumn->colId) {
2,147,483,647✔
1446
        if (cid < 0) {
2,147,483,647✔
1447
          *pColVal = COL_VAL_NULL(pTColumn->colId, pTColumn->type);
3,433,627✔
1448
        } else {
1449
          pColVal->cid = pTColumn->colId;
2,147,483,647✔
1450
          pColVal->value.type = pTColumn->type;
2,147,483,647✔
1451
          pColVal->flag = CV_FLAG_VALUE;
2,147,483,647✔
1452

1453
          if (IS_VAR_DATA_TYPE(pTColumn->type)) {
2,147,483,647!
1454
            int8_t isBlob = IS_STR_DATA_BLOB(pTColumn->type) ? 1 : 0;
1,064,153,233!
1455

1456
            pData += tGetU32v(pData, &pColVal->value.nData);
1,064,153,233!
1457
            if (pColVal->value.nData > 0) {
1,064,153,233!
1458
              pColVal->value.pData = pData;
1,444,020,595✔
1459
            } else {
1460
              pColVal->value.pData = NULL;
×
1461
            }
1462
            if (isBlob == 1) {
1,064,153,233!
1463
              pData += BSE_SEQUECE_SIZE;  // skip seq
×
1464
            }
1465
          } else {
1466
            valueSetDatum(&pColVal->value, pTColumn->type, pData, pTColumn->bytes);
2,147,483,647✔
1467
          }
1468
        }
1469
        return 0;
2,147,483,647✔
1470
      } else if (TABS(cid) < pTColumn->colId) {
2,147,483,647✔
1471
        lidx = mid + 1;
2,147,483,647✔
1472
      } else {
1473
        ridx = mid - 1;
2,147,483,647✔
1474
      }
1475
    }
1476

1477
    *pColVal = COL_VAL_NONE(pTColumn->colId, pTColumn->type);
2,147,483,647✔
1478
  } else {  // Tuple Row
1479
    uint8_t *bitmap = data;
676,284,821✔
1480
    uint8_t *fixed;
1481
    uint8_t *varlen;
1482
    uint8_t  bit;
1483

1484
    if (pRow->flag == HAS_VALUE) {
676,284,821✔
1485
      fixed = bitmap;
636,360,788✔
1486
      bit = BIT_FLG_VALUE;
636,360,788✔
1487
    } else if (pRow->flag == (HAS_NONE | HAS_NULL | HAS_VALUE)) {
39,924,033!
1488
      fixed = BIT2_SIZE(pTSchema->numOfCols - 1) + bitmap;
×
1489
      bit = GET_BIT2(bitmap, iCol - 1);
×
1490
    } else {
1491
      fixed = BIT1_SIZE(pTSchema->numOfCols - 1) + bitmap;
39,924,033✔
1492
      bit = GET_BIT1(bitmap, iCol - 1);
39,924,033✔
1493

1494
      if (pRow->flag == (HAS_NONE | HAS_VALUE)) {
39,924,033✔
1495
        if (bit) bit++;
22,039✔
1496
      } else if (pRow->flag == (HAS_NULL | HAS_VALUE)) {
39,901,994✔
1497
        bit++;
39,645,887✔
1498
      }
1499
    }
1500
    varlen = fixed + pTSchema->flen;
676,284,821✔
1501

1502
    if (bit == BIT_FLG_NONE) {
676,284,821✔
1503
      *pColVal = COL_VAL_NONE(pTColumn->colId, pTColumn->type);
7,701✔
1504
      return 0;
7,701✔
1505
    } else if (bit == BIT_FLG_NULL) {
676,277,120✔
1506
      *pColVal = COL_VAL_NULL(pTColumn->colId, pTColumn->type);
12,311,768✔
1507
      return 0;
12,311,768✔
1508
    }
1509

1510
    pColVal->cid = pTColumn->colId;
663,965,352✔
1511
    pColVal->value.type = pTColumn->type;
663,965,352✔
1512
    pColVal->flag = CV_FLAG_VALUE;
663,965,352✔
1513
    if (IS_VAR_DATA_TYPE(pTColumn->type)) {
663,965,352!
1514
      int8_t isBlob = IS_STR_DATA_BLOB(pTColumn->type) ? 1 : 0;
81,306,648!
1515
      pColVal->value.pData = varlen + *(int32_t *)(fixed + pTColumn->offset);
81,306,648✔
1516
      pColVal->value.pData += tGetU32v(pColVal->value.pData, &pColVal->value.nData);
162,613,296!
1517
      // TODO(yhDeng): support tuple
1518
      //  if (isBlob) pColVal->value.pData += sizeof(uint64_t);  // skip seq
1519
      //  }
1520
    } else {
1521
      valueSetDatum(&pColVal->value, pTColumn->type, fixed + pTColumn->offset, TYPE_BYTES[pTColumn->type]);
582,658,704✔
1522
    }
1523
  }
1524

1525
  return 0;
2,147,483,647✔
1526
}
1527

1528
void tRowDestroy(SRow *pRow) {
1,483,693,328✔
1529
  if (pRow) taosMemoryFree(pRow);
1,483,693,328!
1530
}
1,483,476,921✔
1531

1532
static int32_t tRowPCmprFn(const void *p1, const void *p2) {
549,044,710✔
1533
  SRowKey key1, key2;
1534
  tRowGetKey(*(SRow **)p1, &key1);
1,098,089,420✔
1535
  tRowGetKey(*(SRow **)p2, &key2);
1,089,919,628✔
1536
  return tRowKeyCompare(&key1, &key2);
553,690,973✔
1537
}
1538
static void    tRowPDestroy(SRow **ppRow) { tRowDestroy(*ppRow); }
139,260✔
1539

1540
static SColVal* tRowFindColumnValue(SRowIter *iter, int32_t targetCid) {
92,726✔
1541
  SColVal* pColVal = tRowIterNext(iter);
92,726✔
1542
  while (pColVal != NULL && pColVal->cid < targetCid) {
92,738!
1543
    pColVal = tRowIterNext(iter);
12✔
1544
  }
1545
  return pColVal;
92,726✔
1546
}
1547

1548
static int32_t tRowMergeImpl(SArray *aRowP, STSchema *pTSchema, int32_t iStart, int32_t iEnd, ERowMergeStrategy strategy) {
46,290✔
1549
  int32_t code = 0;
46,290✔
1550

1551
  int32_t    nRow = iEnd - iStart;
46,290✔
1552
  SRowIter **aIter = NULL;
46,290✔
1553
  SArray    *aColVal = NULL;
46,290✔
1554
  SRow      *pRow = NULL;
46,290✔
1555
  uint8_t    hasBlob = 0;
46,290✔
1556
  aIter = taosMemoryCalloc(nRow, sizeof(SRowIter *));
46,290!
1557
  if (aIter == NULL) {
46,290!
1558
    code = terrno;
×
1559
    goto _exit;
×
1560
  }
1561

1562
  for (int32_t i = 0; i < nRow; i++) {
185,550✔
1563
    SRow *pRowT = taosArrayGetP(aRowP, iStart + i);
139,260✔
1564

1565
    code = tRowIterOpen(pRowT, pTSchema, &aIter[i]);
139,260✔
1566
    if (code) goto _exit;
139,260!
1567
  }
1568

1569
  // merge
1570
  aColVal = taosArrayInit(pTSchema->numOfCols, sizeof(SColVal));
46,290✔
1571
  if (aColVal == NULL) {
46,290!
1572
    code = terrno;
×
1573
    goto _exit;
×
1574
  }
1575

1576
  for (int32_t iCol = 0; iCol < pTSchema->numOfCols; iCol++) {
138,992✔
1577
    int32_t targetCid = pTSchema->columns[iCol].colId;
92,702✔
1578
    SColVal *pColVal = NULL;
92,702✔
1579

1580
    switch (strategy) {
92,702✔
1581
      case KEEP_CONSISTENCY:
92,558✔
1582
        if (nRow > 0)
92,558!
1583
          pColVal = tRowFindColumnValue(aIter[nRow - 1], targetCid);
92,558✔
1584
        break;
92,558✔
1585

1586
      default:  // default using PREFER_NON_NULL strategy
144✔
1587
      case PREFER_NON_NULL:
1588
        for (int32_t iRow = nRow - 1; iRow >= 0; --iRow) {
168!
1589
          SColVal *pColValT = tRowFindColumnValue(aIter[iRow], targetCid);
168✔
1590

1591
          if (COL_VAL_IS_VALUE(pColValT)) {
168✔
1592
            pColVal = pColValT;
144✔
1593
            break;
144✔
1594
          } else if (pColVal == NULL) {
24!
1595
            pColVal = pColValT;
24✔
1596
          }
1597
        }
1598
        break;
144✔
1599
    }
1600

1601
    if (pColVal) {
92,702!
1602
      if (taosArrayPush(aColVal, pColVal) == NULL) {
92,702!
1603
        code = terrno;
×
1604
        goto _exit;
×
1605
      }
1606
    }
1607
  }
1608

1609
  // build
1610
  SRowBuildScanInfo sinfo = {0};
46,290✔
1611
  code = tRowBuild(aColVal, pTSchema, &pRow, &sinfo);
46,290✔
1612

1613
  if (code) goto _exit;
46,290!
1614

1615
  taosArrayRemoveBatch(aRowP, iStart, nRow, (FDelete)tRowPDestroy);
46,290✔
1616
  if (taosArrayInsert(aRowP, iStart, &pRow) == NULL) {
46,290!
1617
    code = terrno;
×
1618
    goto _exit;
×
1619
  }
1620

1621
_exit:
46,290✔
1622
  if (aIter) {
46,290!
1623
    for (int32_t i = 0; i < nRow; i++) {
185,550✔
1624
      tRowIterClose(&aIter[i]);
139,260✔
1625
    }
1626
    taosMemoryFree(aIter);
46,290!
1627
  }
1628
  if (aColVal) taosArrayDestroy(aColVal);
46,290!
1629
  if (code) tRowDestroy(pRow);
46,290!
1630
  return code;
46,290✔
1631
}
1632
static int32_t tBlobSetTransferTo(SBlobSet *pSrc, SBlobSet *pDst, SColVal *pVal) {
×
1633
  int32_t code = 0;
×
1634
  int32_t lino = 0;
×
1635
  if (COL_VAL_IS_NULL(pVal) || pVal->value.pData == NULL) {
×
1636
    int8_t type = COL_VAL_IS_NULL(pVal) ? TSDB_DATA_BLOB_NULL_VALUE : TSDB_DATA_BLOB_EMPTY_VALUE;
×
1637
    code = addEmptyItemToBlobSet(pDst, type, NULL);
×
1638
    TAOS_CHECK_GOTO(code, &lino, _error);
×
1639
  } else {
1640
    uint64_t seq = 0;
×
1641
    if (tGetU64(pVal->value.pData, &seq) < 0) {
×
1642
      uError("tBlobSetTransferTo: invalid blob value, seq %p", pVal->value.pData);
×
1643
      return TSDB_CODE_INVALID_PARA;
×
1644
    }
1645

1646
    SBlobItem item = {0};
×
1647
    code = tBlobSetGet(pSrc, seq, &item);
×
1648
    TAOS_CHECK_GOTO(code, &lino, _error);
×
1649

1650
    code = tBlobSetPush(pDst, &item, &seq, 1);
×
1651
    TAOS_CHECK_GOTO(code, &lino, _error);
×
1652
    if (tPutU64(pVal->value.pData, seq) < 0) {
×
1653
      uError("tBlobSetTransferTo: put seq to colVal failed");
×
1654
      return TSDB_CODE_INVALID_PARA;
×
1655
    }
1656
  }
1657

1658
_error:
×
1659
  return code;
×
1660
}
1661

1662
static int32_t tRowRebuildBlob(SArray *aRowP, STSchema *pTSchema, SBlobSet *pBlob) {
×
1663
  int32_t code = 0;
×
1664
  int32_t   nRow = TARRAY_SIZE(aRowP);
×
1665
  int32_t   lino = 0;
×
1666
  SBlobSet *pTempBlob = NULL;
×
1667

1668
  SArray *aColVal = taosArrayInit(pTSchema->numOfCols, sizeof(SColVal));
×
1669
  if (aColVal == NULL) {
×
1670
    TAOS_CHECK_RETURN(code = terrno);
×
1671
  }
1672

1673
  SRowIter **aIter = taosMemoryCalloc(nRow, sizeof(SRowIter *));
×
1674
  if (aIter == NULL) {
×
1675
    TAOS_CHECK_RETURN(code = terrno);
×
1676
  }
1677

1678
  for (int32_t i = 0; i < nRow; i++) {
×
1679
    SRow *pRowT = taosArrayGetP(aRowP, 0 + i);
×
1680
    code = tRowIterOpen(pRowT, pTSchema, &aIter[i]);
×
1681
    TAOS_CHECK_GOTO(code, &lino, _error);
×
1682
  }
1683
  for (int32_t i = 0; i < nRow; i++) {
×
1684
    SColVal *pColVal = tRowIterNext(aIter[i]);
×
1685
    do {
1686
      if (COL_VAL_IS_VALUE(pColVal) || COL_VAL_IS_NULL(pColVal)) {
×
1687
        if (IS_STR_DATA_BLOB(pColVal->value.type)) {
×
1688
          if (taosArrayPush(aColVal, pColVal) == NULL) {
×
1689
            code = terrno;
×
1690
            TAOS_CHECK_GOTO(code, &lino, _error);
×
1691
          }
1692
          break;
×
1693
        }
1694
      }
1695
    } while ((pColVal = tRowIterNext(aIter[i])) != NULL);
×
1696
  }
1697

1698
  code = tBlobSetCreate(pBlob->cap, pBlob->type, &pTempBlob);
×
1699
  TAOS_CHECK_GOTO(code, &lino, _error);
×
1700

1701
  for (int32_t i = 0; i < taosArrayGetSize(aColVal); i++) {
×
1702
    uint64_t seq = 0;
×
1703
    SColVal *pVal = taosArrayGet(aColVal, i);
×
1704
    code = tBlobSetTransferTo(pBlob, pTempBlob, pVal);
×
1705
    TAOS_CHECK_GOTO(code, &lino, _error);
×
1706
  }
1707

1708
  tBlobSetSwap(pBlob, pTempBlob);
×
1709

1710
_error:
×
1711
  if (code != 0) {
×
1712
    uError("tRowRebuildBlob failed at line %d, code %d", code, lino);
×
1713
  }
1714

1715
  if (aIter) {
×
1716
    for (int32_t i = 0; i < nRow; i++) {
×
1717
      tRowIterClose(&aIter[i]);
×
1718
    }
1719
    taosMemoryFree(aIter);
×
1720
  }
1721
  if (aColVal) {
×
1722
    taosArrayDestroy(aColVal);
×
1723
  }
1724
  tBlobSetDestroy(pTempBlob);
×
1725
  return code;
×
1726
}
1727

1728
static int32_t tRowMergeAndRebuildBlob(SArray *aRowP, STSchema *pTSchema, SBlobSet *pBlob) {
×
1729
  int32_t code = 0;
×
1730

1731
  int32_t   lino = 0;
×
1732
  SBlobSet *pTempBlobSet = NULL;
×
1733
  int32_t   size = taosArrayGetSize(aRowP);
×
1734
  if (size <= 1) {
×
1735
    return code;
×
1736
  }
1737
  int32_t colBlobIdx = 0;
×
1738
  for (int32_t i = 0; i < pTSchema->numOfCols; i++) {
×
1739
    if (IS_STR_DATA_BLOB(pTSchema->columns[i].type)) {
×
1740
      colBlobIdx = i;
×
1741
      break;
×
1742
    }
1743
  }
1744

1745
  code = tBlobSetCreate(pBlob->cap, pBlob->type, &pTempBlobSet);
×
1746
  int32_t iStart = 0;
×
1747
  while (iStart < aRowP->size) {
×
1748
    SRowKey key1;
1749
    SRow   *row1 = (SRow *)taosArrayGetP(aRowP, iStart);
×
1750

1751
    tRowGetKey(row1, &key1);
×
1752

1753
    int32_t iEnd = iStart + 1;
×
1754
    while (iEnd < aRowP->size) {
×
1755
      SRowKey key2;
1756
      SRow   *row2 = (SRow *)taosArrayGetP(aRowP, iEnd);
×
1757
      tRowGetKey(row2, &key2);
×
1758

1759
      if (tRowKeyCompare(&key1, &key2) != 0) break;
×
1760

1761
      iEnd++;
×
1762
    }
1763

1764
    if (iEnd - iStart > 1) {
×
1765
      code = tRowMergeWithBlobImpl(aRowP, pTSchema, pBlob, pTempBlobSet, iStart, iEnd, 0);
×
1766
      TAOS_CHECK_GOTO(code, &lino, _error);
×
1767
    } else {
1768
      SColVal colVal = {0};
×
1769
      code = tRowGet(row1, pTSchema, colBlobIdx, &colVal);
×
1770
      TAOS_CHECK_GOTO(code, &lino, _error);
×
1771

1772
      code = tBlobSetTransferTo(pBlob, pTempBlobSet, &colVal);
×
1773
      TAOS_CHECK_GOTO(code, &lino, _error);
×
1774
    }
1775
    // the array is also changing, so the iStart just ++ instead of iEnd
1776
    iStart++;
×
1777
  }
1778
_error:
×
1779
  tBlobSetSwap(pBlob, pTempBlobSet);
×
1780
  tBlobSetDestroy(pTempBlobSet);
×
1781
  return code;
×
1782
}
1783

1784
static int32_t tRowMergeWithBlobImpl(SArray *aRowP, STSchema *pTSchema, SBlobSet *pBlob, SBlobSet *pDstBlob,
×
1785
                                     int32_t iStart, int32_t iEnd, int8_t flag) {
1786
  int32_t code = 0;
×
1787
  int32_t    lino = 0;
×
1788
  int32_t    nRow = iEnd - iStart;
×
1789
  SRowIter **aIter = NULL;
×
1790
  SArray    *aColVal = NULL;
×
1791
  SRow      *pRow = NULL;
×
1792

1793
  aColVal = taosArrayInit(pTSchema->numOfCols, sizeof(SColVal));
×
1794
  if (aColVal == NULL) {
×
1795
    code = terrno;
×
1796
    TAOS_CHECK_GOTO(code, &lino, _exit);
×
1797
  }
1798

1799
  aIter = taosMemoryCalloc(nRow, sizeof(SRowIter *));
×
1800
  if (aIter == NULL) {
×
1801
    code = terrno;
×
1802
    TAOS_CHECK_GOTO(code, &lino, _exit);
×
1803
  }
1804

1805
  for (int32_t i = 0; i < nRow; i++) {
×
1806
    SRow *pRowT = taosArrayGetP(aRowP, iStart + i);
×
1807
    code = tRowIterOpen(pRowT, pTSchema, &aIter[i]);
×
1808
    TAOS_CHECK_GOTO(code, &lino, _exit);
×
1809
  }
1810
  // merge
1811

1812
  for (int32_t iCol = 0; iCol < pTSchema->numOfCols; iCol++) {
×
1813
    SColVal  *pColVal = NULL;
×
1814
    STColumn *pCol = pTSchema->columns + iCol;
×
1815

1816
    for (int32_t iRow = nRow - 1; iRow >= 0; --iRow) {
×
1817
      SColVal *pColValT = tRowIterNext(aIter[iRow]);
×
1818
      while (pColValT->cid < pTSchema->columns[iCol].colId) {
×
1819
        pColValT = tRowIterNext(aIter[iRow]);
×
1820
      }
1821
      // todo: take strategy according to the flag
1822
      if (COL_VAL_IS_VALUE(pColValT)) {
×
1823
        pColVal = pColValT;
×
1824
        break;
×
1825
      } else if (COL_VAL_IS_NULL(pColValT)) {
×
1826
        if (pColVal == NULL) {
×
1827
          pColVal = pColValT;
×
1828
        }
1829
      }
1830
    }
1831

1832
    if (pColVal) {
×
1833
      if (taosArrayPush(aColVal, pColVal) == NULL) {
×
1834
        code = terrno;
×
1835
        TAOS_CHECK_GOTO(code, &lino, _exit);
×
1836
      }
1837
    }
1838
  }
1839

1840
  // build
1841

1842
  SRowBuildScanInfo sinfo = {.hasBlob = 1};
×
1843
  code = tRowBuildWithBlob2(aColVal, pTSchema, &pRow, pBlob, pDstBlob, &sinfo);
×
1844
  TAOS_CHECK_GOTO(code, &lino, _exit);
×
1845

1846
  taosArrayRemoveBatch(aRowP, iStart, nRow, (FDelete)tRowPDestroy);
×
1847
  if (taosArrayInsert(aRowP, iStart, &pRow) == NULL) {
×
1848
    code = terrno;
×
1849
    TAOS_CHECK_GOTO(code, &lino, _exit);
×
1850
  }
1851

1852
_exit:
×
1853
  if (aIter) {
×
1854
    for (int32_t i = 0; i < nRow; i++) {
×
1855
      tRowIterClose(&aIter[i]);
×
1856
    }
1857
    taosMemoryFree(aIter);
×
1858
  }
1859
  if (aColVal) taosArrayDestroy(aColVal);
×
1860
  if (code) tRowDestroy(pRow);
×
1861

1862
  return code;
×
1863
}
1864

1865
int32_t tRowSort(SArray *aRowP) {
78,156✔
1866
  if (TARRAY_SIZE(aRowP) <= 1) return 0;
78,156✔
1867
  int32_t code = taosArrayMSort(aRowP, tRowPCmprFn);
78,154✔
1868
  if (code != TSDB_CODE_SUCCESS) {
78,156!
1869
    uError("taosArrayMSort failed caused by %d", code);
×
1870
  }
1871
  return code;
78,156✔
1872
}
1873

1874
int32_t tRowMerge(SArray *aRowP, STSchema *pTSchema, ERowMergeStrategy strategy) {
78,177✔
1875
  int32_t code = 0;
78,177✔
1876

1877
  int32_t iStart = 0;
78,177✔
1878
  while (iStart < aRowP->size) {
75,699,647✔
1879
    SRowKey key1;
1880
    SRow   *row1 = (SRow *)taosArrayGetP(aRowP, iStart);
75,677,715✔
1881

1882
    tRowGetKey(row1, &key1);
151,218,290✔
1883

1884
    int32_t iEnd = iStart + 1;
75,621,283✔
1885
    while (iEnd < aRowP->size) {
75,676,730✔
1886
      SRowKey key2;
1887
      SRow   *row2 = (SRow *)taosArrayGetP(aRowP, iEnd);
75,633,281✔
1888
      tRowGetKey(row2, &key2);
151,197,256✔
1889

1890
      if (tRowKeyCompare(&key1, &key2) != 0) break;
75,633,468✔
1891

1892
      iEnd++;
55,447✔
1893
    }
1894

1895
    if (iEnd - iStart > 1) {
75,621,470✔
1896
      code = tRowMergeImpl(aRowP, pTSchema, iStart, iEnd, strategy);
46,290✔
1897
      if (code) return code;
46,290!
1898
    }
1899

1900
    // the array is also changing, so the iStart just ++ instead of iEnd
1901
    iStart++;
75,621,470✔
1902
  }
1903

1904
  return code;
21,932✔
1905
}
1906

1907
int32_t tRowSortWithBlob(SArray *aRowP, STSchema *pTSchema, SBlobSet *pBlobSet) {
×
1908
  if (TARRAY_SIZE(aRowP) <= 1) return 0;
×
1909
  int32_t code = taosArrayMSort(aRowP, tRowPCmprFn);
×
1910
  if (code != TSDB_CODE_SUCCESS) {
×
1911
    uError("taosArrayMSort failed caused by %d", code);
×
1912
    return code;
×
1913
  }
1914
  return code;
×
1915
}
1916

1917
int8_t tRowNeedDoMerge(SArray *aRowP) {
×
1918
  if (aRowP == NULL || aRowP->size <= 1) return 0;
×
1919

1920
  int8_t  doMerge = 0;
×
1921
  int32_t iStart = 0;
×
1922
  while (iStart < aRowP->size) {
×
1923
    SRowKey key1;
1924
    SRow   *row1 = (SRow *)taosArrayGetP(aRowP, iStart);
×
1925

1926
    tRowGetKey(row1, &key1);
×
1927

1928
    int32_t iEnd = iStart + 1;
×
1929
    while (iEnd < aRowP->size) {
×
1930
      SRowKey key2;
1931
      SRow   *row2 = (SRow *)taosArrayGetP(aRowP, iEnd);
×
1932
      tRowGetKey(row2, &key2);
×
1933

1934
      if (tRowKeyCompare(&key1, &key2) != 0) break;
×
1935

1936
      iEnd++;
×
1937
    }
1938

1939
    if (iEnd - iStart > 1) {
×
1940
      doMerge = 1;
×
1941
      break;
×
1942
    }
1943
    iStart++;
×
1944
  }
1945
  return doMerge;
×
1946
}
1947

1948
int32_t tRowMergeWithBlob(SArray *aRowP, STSchema *pTSchema, SBlobSet *pBlobSet, int8_t flag) {
×
1949
  int32_t code = 0;
×
1950

1951
  int8_t doMerge = tRowNeedDoMerge(aRowP);
×
1952
  if (!doMerge) {
×
1953
    code = tRowRebuildBlob(aRowP, pTSchema, pBlobSet);
×
1954
  } else {
1955
    code = tRowRebuildBlob(aRowP, pTSchema, pBlobSet);
×
1956
    TAOS_CHECK_RETURN(code);
×
1957

1958
    code = tRowMergeAndRebuildBlob(aRowP, pTSchema, pBlobSet);
×
1959
  }
1960
  return code;
×
1961
}
1962

1963
// SRowIter ========================================
1964
struct SRowIter {
1965
  SRow     *pRow;
1966
  STSchema *pTSchema;
1967

1968
  int32_t iTColumn;
1969
  union {
1970
    struct {  // kv
1971
      int32_t iCol;
1972
      SKVIdx *pIdx;
1973
    };
1974
    struct {  // tuple
1975
      uint8_t *pb;
1976
      uint8_t *pf;
1977
    };
1978
  };
1979
  uint8_t *pv;
1980
  SColVal  cv;
1981
};
1982

1983
int32_t tRowIterOpen(SRow *pRow, STSchema *pTSchema, SRowIter **ppIter) {
558,701✔
1984
  if (!(pRow->sver == pTSchema->version)) return TSDB_CODE_INVALID_PARA;
558,701!
1985

1986
  int32_t code = 0;
558,701✔
1987

1988
  SRowIter *pIter = taosMemoryCalloc(1, sizeof(*pIter));
558,701!
1989
  if (pIter == NULL) {
559,388!
1990
    code = terrno;
×
1991
    goto _exit;
×
1992
  }
1993

1994
  pIter->pRow = pRow;
559,388✔
1995
  pIter->pTSchema = pTSchema;
559,388✔
1996
  pIter->iTColumn = 0;
559,388✔
1997

1998
  if (pRow->flag == HAS_NONE || pRow->flag == HAS_NULL) goto _exit;
559,388✔
1999

2000
  uint8_t         *data = pRow->data;
556,081✔
2001
  SPrimaryKeyIndex index;
2002
  for (int32_t i = 0; i < pRow->numOfPKs; i++) {
583,812✔
2003
    data += tGetPrimaryKeyIndex(data, &index);
27,742✔
2004
  }
2005

2006
  if (pRow->flag >> 4) {
556,070✔
2007
    pIter->iCol = 0;
412,838✔
2008
    pIter->pIdx = (SKVIdx *)data;
412,838✔
2009
    if (pRow->flag & KV_FLG_LIT) {
412,838✔
2010
      pIter->pv = pIter->pIdx->idx + pIter->pIdx->nCol;
407,130✔
2011
    } else if (pRow->flag & KV_FLG_MID) {
5,708✔
2012
      pIter->pv = pIter->pIdx->idx + (pIter->pIdx->nCol << 1);  // * sizeof(uint16_t)
5,700✔
2013
    } else {
2014
      pIter->pv = pIter->pIdx->idx + (pIter->pIdx->nCol << 2);  // * sizeof(uint32_t)
8✔
2015
    }
2016
  } else {
2017
    switch (pRow->flag) {
143,232!
2018
      case (HAS_NULL | HAS_NONE):
×
2019
        pIter->pb = data;
×
2020
        break;
×
2021
      case HAS_VALUE:
142,089✔
2022
        pIter->pf = data;
142,089✔
2023
        pIter->pv = pIter->pf + pTSchema->flen;
142,089✔
2024
        break;
142,089✔
2025
      case (HAS_VALUE | HAS_NONE):
1,135✔
2026
      case (HAS_VALUE | HAS_NULL):
2027
        pIter->pb = data;
1,135✔
2028
        pIter->pf = data + BIT1_SIZE(pTSchema->numOfCols - 1);
1,135✔
2029
        pIter->pv = pIter->pf + pTSchema->flen;
1,135✔
2030
        break;
1,135✔
2031
      case (HAS_VALUE | HAS_NULL | HAS_NONE):
×
2032
        pIter->pb = data;
×
2033
        pIter->pf = data + BIT2_SIZE(pTSchema->numOfCols - 1);
×
2034
        pIter->pv = pIter->pf + pTSchema->flen;
×
2035
        break;
×
2036
      default:
8✔
2037
        break;
8✔
2038
    }
2039
  }
2040

2041
_exit:
559,377✔
2042
  if (code) {
559,377!
2043
    *ppIter = NULL;
×
2044
  } else {
2045
    *ppIter = pIter;
559,377✔
2046
  }
2047
  return code;
559,377✔
2048
}
2049

2050
void tRowIterClose(SRowIter **ppIter) {
558,178✔
2051
  SRowIter *pIter = *ppIter;
558,178✔
2052
  if (pIter) {
558,178✔
2053
    taosMemoryFree(pIter);
558,177!
2054
  }
2055
  *ppIter = NULL;
559,409✔
2056
}
559,409✔
2057

2058
SColVal *tRowIterNext(SRowIter *pIter) {
10,718,411✔
2059
  if (pIter->iTColumn >= pIter->pTSchema->numOfCols) {
10,718,411✔
2060
    return NULL;
410,165✔
2061
  }
2062

2063
  STColumn *pTColumn = pIter->pTSchema->columns + pIter->iTColumn;
10,308,246✔
2064

2065
  // timestamp
2066
  if (0 == pIter->iTColumn) {
10,308,246✔
2067
    pIter->cv.cid = pTColumn->colId;
466,308✔
2068
    pIter->cv.value.type = pTColumn->type;
466,308✔
2069
    pIter->cv.flag = CV_FLAG_VALUE;
466,308✔
2070
    VALUE_SET_TRIVIAL_DATUM(&pIter->cv.value, pIter->pRow->ts);
466,308✔
2071
    goto _exit;
466,308✔
2072
  }
2073

2074
  if (pIter->pRow->flag == HAS_NONE) {
9,841,938✔
2075
    pIter->cv = COL_VAL_NONE(pTColumn->colId, pTColumn->type);
55,812✔
2076
    goto _exit;
55,812✔
2077
  }
2078

2079
  if (pIter->pRow->flag == HAS_NULL) {
9,786,126✔
2080
    pIter->cv = COL_VAL_NULL(pTColumn->colId, pTColumn->type);
3,043✔
2081
    goto _exit;
3,043✔
2082
  }
2083

2084
  if (pIter->pRow->flag >> 4) {  // KV
9,783,083✔
2085
    if (pIter->iCol < pIter->pIdx->nCol) {
9,735,974✔
2086
      uint8_t *pData;
2087

2088
      if (pIter->pRow->flag & KV_FLG_LIT) {
5,597,004✔
2089
        pData = pIter->pv + ((uint8_t *)pIter->pIdx->idx)[pIter->iCol];
5,426,052✔
2090
      } else if (pIter->pRow->flag & KV_FLG_MID) {
170,952!
2091
        pData = pIter->pv + ((uint16_t *)pIter->pIdx->idx)[pIter->iCol];
171,000✔
2092
      } else {
2093
        pData = pIter->pv + ((uint32_t *)pIter->pIdx->idx)[pIter->iCol];
×
2094
      }
2095

2096
      int16_t cid;
2097
      pData += tGetI16v(pData, &cid);
5,597,004✔
2098

2099
      if (TABS(cid) == pTColumn->colId) {
5,597,004!
2100
        if (cid < 0) {
5,599,867✔
2101
          pIter->cv = COL_VAL_NULL(pTColumn->colId, pTColumn->type);
4,476✔
2102
        } else {
2103
          pIter->cv.cid = pTColumn->colId;
5,595,391✔
2104
          pIter->cv.value.type = pTColumn->type;
5,595,391✔
2105
          pIter->cv.flag = CV_FLAG_VALUE;
5,595,391✔
2106

2107
          if (IS_VAR_DATA_TYPE(pTColumn->type)) {
5,595,391!
2108
            pData += tGetU32v(pData, &pIter->cv.value.nData);
875,772!
2109
            if (pIter->cv.value.nData > 0) {
875,772!
2110
              pIter->cv.value.pData = pData;
884,115✔
2111
            } else {
2112
              pIter->cv.value.pData = NULL;
×
2113
              // if (IS_STR_DATA_BLOB(pTColumn->type)) {
2114
              //   pIter->cv.value.pData = pData;
2115
              // }
2116
            }
2117
          } else {
2118
            valueSetDatum(&pIter->cv.value, pTColumn->type, pData, pTColumn->bytes);
4,719,619✔
2119
          }
2120
        }
2121

2122
        pIter->iCol++;
5,615,396✔
2123
        goto _exit;
5,615,396✔
2124
      } else if (TABS(cid) > pTColumn->colId) {
×
2125
        pIter->cv = COL_VAL_NONE(pTColumn->colId, pTColumn->type);
24✔
2126
        goto _exit;
24✔
2127
      } else {
2128
        uError("unexpected column id %d, %d", cid, pTColumn->colId);
×
2129
        goto _exit;
×
2130
      }
2131
    } else {
2132
      pIter->cv = COL_VAL_NONE(pTColumn->colId, pTColumn->type);
4,138,970✔
2133
      goto _exit;
4,138,970✔
2134
    }
2135
  } else {  // Tuple
2136
    uint8_t bv = BIT_FLG_VALUE;
47,109✔
2137
    if (pIter->pb) {
47,109✔
2138
      switch (pIter->pRow->flag) {
4,898!
2139
        case (HAS_NULL | HAS_NONE):
×
2140
          bv = GET_BIT1(pIter->pb, pIter->iTColumn - 1);
×
2141
          break;
×
2142
        case (HAS_VALUE | HAS_NONE):
12✔
2143
          bv = GET_BIT1(pIter->pb, pIter->iTColumn - 1);
12✔
2144
          if (bv) bv++;
12!
2145
          break;
12✔
2146
        case (HAS_VALUE | HAS_NULL):
4,886✔
2147
          bv = GET_BIT1(pIter->pb, pIter->iTColumn - 1) + 1;
4,886✔
2148
          break;
4,886✔
2149
        case (HAS_VALUE | HAS_NULL | HAS_NONE):
×
2150
          bv = GET_BIT2(pIter->pb, pIter->iTColumn - 1);
×
2151
          break;
×
2152
        default:
×
2153
          break;
×
2154
      }
2155

2156
      if (bv == BIT_FLG_NONE) {
4,898!
2157
        pIter->cv = COL_VAL_NONE(pTColumn->colId, pTColumn->type);
×
2158
        goto _exit;
×
2159
      } else if (bv == BIT_FLG_NULL) {
4,898✔
2160
        pIter->cv = COL_VAL_NULL(pTColumn->colId, pTColumn->type);
549✔
2161
        goto _exit;
549✔
2162
      }
2163
    }
2164

2165
    pIter->cv.cid = pTColumn->colId;
46,560✔
2166
    pIter->cv.value.type = pTColumn->type;
46,560✔
2167
    pIter->cv.flag = CV_FLAG_VALUE;
46,560✔
2168
    if (IS_VAR_DATA_TYPE(pTColumn->type)) {
20,046!
2169
      uint8_t *pData = pIter->pv + *(int32_t *)(pIter->pf + pTColumn->offset);
×
2170
      pData += tGetU32v(pData, &pIter->cv.value.nData);
×
2171
      if (pIter->cv.value.nData > 0) {
×
2172
        pIter->cv.value.pData = pData;
1,005✔
2173
      } else {
2174
        pIter->cv.value.pData = NULL;
×
2175
        // if (IS_STR_DATA_BLOB(pTColumn->type)) {
2176
        //   pIter->cv.value.pData = pData;
2177
        // }
2178
      }
2179
    } else {
2180
      valueSetDatum(&pIter->cv.value, pTColumn->type, pIter->pf + pTColumn->offset, TYPE_BYTES[pTColumn->type]);
73,074✔
2181
    }
2182
    goto _exit;
74,079✔
2183
  }
2184

2185
_exit:
10,354,181✔
2186
  pIter->iTColumn++;
10,354,181✔
2187
  return &pIter->cv;
10,354,181✔
2188
}
2189

2190
static int32_t tRowNoneUpsertColData(SColData *aColData, int32_t nColData, int32_t flag) {
7,913✔
2191
  int32_t code = 0;
7,913✔
2192

2193
  if (flag) return code;
7,913✔
2194

2195
  for (int32_t iColData = 0; iColData < nColData; iColData++) {
66,025✔
2196
    code = tColDataAppendValueImpl[aColData[iColData].flag][CV_FLAG_NONE](&aColData[iColData], NULL, 0);
61,294✔
2197
    if (code) return code;
61,280!
2198
  }
2199

2200
  return code;
4,731✔
2201
}
2202
static int32_t tRowNullUpsertColData(SColData *aColData, int32_t nColData, STSchema *pSchema, int32_t flag) {
167,564✔
2203
  int32_t code = 0;
167,564✔
2204

2205
  int32_t   iColData = 0;
167,564✔
2206
  SColData *pColData = &aColData[iColData];
167,564✔
2207
  int32_t   iTColumn = 1;
167,564✔
2208
  STColumn *pTColumn = &pSchema->columns[iTColumn];
167,564✔
2209

2210
  while (pColData) {
611,200✔
2211
    if (pTColumn) {
443,635!
2212
      if (pTColumn->colId == pColData->cid) {  // NULL
443,635✔
2213
        if (flag == 0) {
443,625✔
2214
          code = tColDataAppendValueImpl[pColData->flag][CV_FLAG_NULL](pColData, NULL, 0);
443,493✔
2215
        } else {
2216
          code = tColDataUpdateValueImpl[pColData->flag][CV_FLAG_NULL](pColData, NULL, 0, flag > 0);
132✔
2217
        }
2218
        if (code) goto _exit;
443,626!
2219

2220
        pColData = (++iColData < nColData) ? &aColData[iColData] : NULL;
443,626✔
2221
        pTColumn = (++iTColumn < pSchema->numOfCols) ? &pSchema->columns[iTColumn] : NULL;
443,626✔
2222
      } else if (pTColumn->colId > pColData->cid) {  // NONE
10!
2223
        if (flag == 0 && (code = tColDataAppendValueImpl[pColData->flag][CV_FLAG_NONE](pColData, NULL, 0))) goto _exit;
×
2224
        pColData = (++iColData < nColData) ? &aColData[iColData] : NULL;
×
2225
      } else {
2226
        pTColumn = (++iTColumn < pSchema->numOfCols) ? &pSchema->columns[iTColumn] : NULL;
10!
2227
      }
2228
    } else {  // NONE
2229
      if (flag == 0 && (code = tColDataAppendValueImpl[pColData->flag][CV_FLAG_NONE](pColData, NULL, 0))) goto _exit;
×
2230
      pColData = (++iColData < nColData) ? &aColData[iColData] : NULL;
×
2231
    }
2232
  }
2233

2234
_exit:
167,565✔
2235
  return code;
167,565✔
2236
}
2237
static int32_t tRowTupleUpsertColData(SRow *pRow, STSchema *pTSchema, SColData *aColData, int32_t nColData,
1,571,910,908✔
2238
                                      int32_t flag) {
2239
  int32_t code = 0;
1,571,910,908✔
2240

2241
  int32_t   iColData = 0;
1,571,910,908✔
2242
  SColData *pColData = &aColData[iColData];
1,571,910,908✔
2243
  int32_t   iTColumn = 1;
1,571,910,908✔
2244
  STColumn *pTColumn = &pTSchema->columns[iTColumn];
1,571,910,908✔
2245

2246
  uint8_t         *pb = NULL, *pf = NULL, *pv = NULL;
1,571,910,908✔
2247
  SPrimaryKeyIndex index;
2248
  uint8_t         *data = pRow->data;
1,571,910,908✔
2249
  for (int32_t i = 0; i < pRow->numOfPKs; i++) {
2,040,708,203✔
2250
    data += tGetPrimaryKeyIndex(data, &index);
469,067,761✔
2251
  }
2252

2253
  switch (pRow->flag) {
1,571,640,442!
2254
    case HAS_VALUE:
1,556,012,218✔
2255
      pf = data;  // TODO: fix here
1,556,012,218✔
2256
      pv = pf + pTSchema->flen;
1,556,012,218✔
2257
      break;
1,556,012,218✔
2258
    case (HAS_NULL | HAS_NONE):
1,296✔
2259
      pb = data;
1,296✔
2260
      break;
1,296✔
2261
    case (HAS_VALUE | HAS_NONE):
16,434,447✔
2262
    case (HAS_VALUE | HAS_NULL):
2263
      pb = data;
16,434,447✔
2264
      pf = pb + BIT1_SIZE(pTSchema->numOfCols - 1);
16,434,447✔
2265
      pv = pf + pTSchema->flen;
16,434,447✔
2266
      break;
16,434,447✔
2267
    case (HAS_VALUE | HAS_NULL | HAS_NONE):
×
2268
      pb = data;
×
2269
      pf = pb + BIT2_SIZE(pTSchema->numOfCols - 1);
×
2270
      pv = pf + pTSchema->flen;
×
2271
      break;
×
2272
    default:
×
2273
      return TSDB_CODE_INVALID_DATA_FMT;
×
2274
  }
2275

2276
  while (pColData) {
2,147,483,647✔
2277
    if (pTColumn) {
2,147,483,647✔
2278
      if (pTColumn->colId == pColData->cid) {
2,147,483,647✔
2279
        if (!(pTColumn->type == pColData->type)) {
2,147,483,647!
2280
          return TSDB_CODE_INVALID_PARA;
×
2281
        }
2282
        if (pb) {
2,147,483,647✔
2283
          uint8_t bv;
2284
          switch (pRow->flag) {
70,161,484!
2285
            case (HAS_NULL | HAS_NONE):
25,920✔
2286
              bv = GET_BIT1(pb, iTColumn - 1);
25,920✔
2287
              break;
25,920✔
2288
            case (HAS_VALUE | HAS_NONE):
331,625✔
2289
              bv = GET_BIT1(pb, iTColumn - 1);
331,625✔
2290
              if (bv) bv++;
331,625✔
2291
              break;
331,625✔
2292
            case (HAS_VALUE | HAS_NULL):
69,805,457✔
2293
              bv = GET_BIT1(pb, iTColumn - 1) + 1;
69,805,457✔
2294
              break;
69,805,457✔
2295
            case (HAS_VALUE | HAS_NULL | HAS_NONE):
×
2296
              bv = GET_BIT2(pb, iTColumn - 1);
×
2297
              break;
×
2298
            default:
×
2299
              return TSDB_CODE_INVALID_DATA_FMT;
×
2300
          }
2301

2302
          if (bv == BIT_FLG_NONE) {
70,163,002✔
2303
            if (flag == 0 && (code = tColDataAppendValueImpl[pColData->flag][CV_FLAG_NONE](pColData, NULL, 0)))
53,358!
2304
              goto _exit;
×
2305
            goto _continue;
53,348✔
2306
          } else if (bv == BIT_FLG_NULL) {
70,109,644✔
2307
            if (flag == 0) {
19,380,103✔
2308
              code = tColDataAppendValueImpl[pColData->flag][CV_FLAG_NULL](pColData, NULL, 0);
19,377,762✔
2309
            } else {
2310
              code = tColDataUpdateValueImpl[pColData->flag][CV_FLAG_NULL](pColData, NULL, 0, flag > 0);
2,341✔
2311
            }
2312
            if (code) goto _exit;
19,384,701!
2313
            goto _continue;
19,384,701✔
2314
          }
2315
        }
2316

2317
        if (IS_VAR_DATA_TYPE(pColData->type)) {
2,147,483,647!
2318
          uint8_t *pData = pv + *(int32_t *)(pf + pTColumn->offset);
78,666,238!
2319
          uint32_t nData;
2320
          pData += tGetU32v(pData, &nData);
78,666,238✔
2321
          if (flag == 0) {
78,666,238!
2322
            code = tColDataAppendValueImpl[pColData->flag][CV_FLAG_VALUE](pColData, pData, nData);
84,990,934✔
2323
          } else {
2324
            code = tColDataUpdateValueImpl[pColData->flag][CV_FLAG_VALUE](pColData, pData, nData, flag > 0);
×
2325
          }
2326
          if (code) goto _exit;
85,558,671!
2327
        } else {
2328
          if (flag == 0) {
2,147,483,647✔
2329
            code = tColDataAppendValueImpl[pColData->flag][CV_FLAG_VALUE](pColData, pf + pTColumn->offset,
2,147,483,647✔
2330
                                                                          TYPE_BYTES[pColData->type]);
2,147,483,647✔
2331
          } else {
2332
            code = tColDataUpdateValueImpl[pColData->flag][CV_FLAG_VALUE](pColData, pf + pTColumn->offset,
5,013,463✔
2333
                                                                          TYPE_BYTES[pColData->type], flag > 0);
5,013,463✔
2334
          }
2335
          if (code) goto _exit;
2,147,483,647!
2336
        }
2337

2338
      _continue:
2,147,483,647✔
2339
        pColData = (++iColData < nColData) ? &aColData[iColData] : NULL;
2,147,483,647✔
2340
        pTColumn = (++iTColumn < pTSchema->numOfCols) ? &pTSchema->columns[iTColumn] : NULL;
2,147,483,647✔
2341
      } else if (pTColumn->colId > pColData->cid) {  // NONE
1,020,890!
2342
        if (flag == 0 && (code = tColDataAppendValueImpl[pColData->flag][CV_FLAG_NONE](pColData, NULL, 0))) goto _exit;
×
2343
        pColData = (++iColData < nColData) ? &aColData[iColData] : NULL;
×
2344
      } else {
2345
        pTColumn = (++iTColumn < pTSchema->numOfCols) ? &pTSchema->columns[iTColumn] : NULL;
1,020,890✔
2346
      }
2347
    } else {
2348
      if (flag == 0 && (code = tColDataAppendValueImpl[pColData->flag][CV_FLAG_NONE](pColData, NULL, 0))) goto _exit;
20,370!
2349
      pColData = (++iColData < nColData) ? &aColData[iColData] : NULL;
20,370✔
2350
    }
2351
  }
2352

2353
_exit:
1,572,198,740✔
2354
  return code;
1,572,198,740✔
2355
}
2356
static int32_t tRowKVUpsertColData(SRow *pRow, STSchema *pTSchema, SColData *aColData, int32_t nColData, int32_t flag) {
4,284,501✔
2357
  int32_t code = 0;
4,284,501✔
2358

2359
  uint8_t  *pv = NULL;
4,284,501✔
2360
  int32_t   iColData = 0;
4,284,501✔
2361
  SColData *pColData = &aColData[iColData];
4,284,501✔
2362
  int32_t   iTColumn = 1;
4,284,501✔
2363
  STColumn *pTColumn = &pTSchema->columns[iTColumn];
4,284,501✔
2364
  int32_t   iCol = 0;
4,284,501✔
2365

2366
  // primary keys
2367
  uint8_t         *data = pRow->data;
4,284,501✔
2368
  SPrimaryKeyIndex index;
2369
  for (int32_t i = 0; i < pRow->numOfPKs; i++) {
4,695,696✔
2370
    data += tGetPrimaryKeyIndex(data, &index);
411,246✔
2371
  }
2372

2373
  SKVIdx *pKVIdx = (SKVIdx *)data;
4,284,450✔
2374
  if (pRow->flag & KV_FLG_LIT) {
4,284,450✔
2375
    pv = pKVIdx->idx + pKVIdx->nCol;
4,229,885✔
2376
  } else if (pRow->flag & KV_FLG_MID) {
54,565!
2377
    pv = pKVIdx->idx + (pKVIdx->nCol << 1);
55,582✔
2378
  } else if (pRow->flag & KV_FLG_BIG) {
×
2379
    pv = pKVIdx->idx + (pKVIdx->nCol << 2);
×
2380
  } else {
2381
    return TSDB_CODE_INVALID_PARA;
×
2382
  }
2383

2384
  while (pColData) {
61,425,411✔
2385
    if (pTColumn) {
57,168,795✔
2386
      if (pTColumn->colId == pColData->cid) {
57,168,763✔
2387
        while (iCol < pKVIdx->nCol) {
57,127,187✔
2388
          uint8_t *pData;
2389
          if (pRow->flag & KV_FLG_LIT) {
47,393,311✔
2390
            pData = pv + ((uint8_t *)pKVIdx->idx)[iCol];
45,790,057✔
2391
          } else if (pRow->flag & KV_FLG_MID) {
1,603,254!
2392
            pData = pv + ((uint16_t *)pKVIdx->idx)[iCol];
1,603,339✔
2393
          } else if (pRow->flag & KV_FLG_BIG) {
×
2394
            pData = pv + ((uint32_t *)pKVIdx->idx)[iCol];
×
2395
          } else {
2396
            return TSDB_CODE_INVALID_DATA_FMT;
×
2397
          }
2398

2399
          int16_t cid;
2400
          pData += tGetI16v(pData, &cid);
47,393,396✔
2401

2402
          if (TABS(cid) == pTColumn->colId) {
47,393,396✔
2403
            if (cid < 0) {
28,589,630✔
2404
              if (flag == 0) {
5,307,743✔
2405
                code = tColDataAppendValueImpl[pColData->flag][CV_FLAG_NULL](pColData, NULL, 0);
5,306,396✔
2406
              } else {
2407
                code = tColDataUpdateValueImpl[pColData->flag][CV_FLAG_NULL](pColData, NULL, 0, flag > 0);
1,347✔
2408
              }
2409
              if (code) goto _exit;
4,964,624!
2410
            } else {
2411
              uint32_t nData;
2412
              if (IS_VAR_DATA_TYPE(pTColumn->type)) {
23,281,887!
2413
                pData += tGetU32v(pData, &nData);
4,550,348✔
2414
              } else {
2415
                nData = 0;
18,731,539✔
2416
              }
2417
              if (flag == 0) {
23,281,887!
2418
                code = tColDataAppendValueImpl[pColData->flag][CV_FLAG_VALUE](pColData, pData, nData);
23,339,436✔
2419
              } else {
2420
                code = tColDataUpdateValueImpl[pColData->flag][CV_FLAG_VALUE](pColData, pData, nData, flag > 0);
×
2421
              }
2422
              if (code) goto _exit;
23,597,080!
2423
            }
2424
            iCol++;
28,561,704✔
2425
            goto _continue;
28,561,704✔
2426
          } else if (TABS(cid) > pTColumn->colId) {  // NONE
18,803,766!
2427
            break;
18,816,017✔
2428
          } else {
2429
            iCol++;
×
2430
          }
2431
        }
2432

2433
        if (flag == 0 && (code = tColDataAppendValueImpl[pColData->flag][CV_FLAG_NONE](pColData, NULL, 0))) goto _exit;
28,549,893!
2434

2435
      _continue:
28,548,883✔
2436
        pColData = (++iColData < nColData) ? &aColData[iColData] : NULL;
57,110,587✔
2437
        pTColumn = (++iTColumn < pTSchema->numOfCols) ? &pTSchema->columns[iTColumn] : NULL;
57,110,587✔
2438
      } else if (pTColumn->colId > pColData->cid) {
29,325!
2439
        if (flag == 0 && (code = tColDataAppendValueImpl[pColData->flag][CV_FLAG_NONE](pColData, NULL, 0))) goto _exit;
×
2440
        pColData = (++iColData < nColData) ? &aColData[iColData] : NULL;
×
2441
      } else {
2442
        pTColumn = (++iTColumn < pTSchema->numOfCols) ? &pTSchema->columns[iTColumn] : NULL;
29,325!
2443
      }
2444
    } else {
2445
      if (flag == 0 && (code = tColDataAppendValueImpl[pColData->flag][CV_FLAG_NONE](pColData, NULL, 0))) goto _exit;
32!
2446
      pColData = (++iColData < nColData) ? &aColData[iColData] : NULL;
32!
2447
    }
2448
  }
2449

2450
_exit:
4,256,616✔
2451
  return code;
4,256,616✔
2452
}
2453
/* flag > 0: forward update
2454
 * flag == 0: append
2455
 * flag < 0: backward update
2456
 */
2457
int32_t tRowUpsertColData(SRow *pRow, STSchema *pTSchema, SColData *aColData, int32_t nColData, int32_t flag) {
1,576,600,104✔
2458
  if (!(pRow->sver == pTSchema->version)) return TSDB_CODE_INVALID_PARA;
1,576,600,104!
2459
  if (!(nColData > 0)) return TSDB_CODE_INVALID_PARA;
1,576,600,104!
2460

2461
  if (pRow->flag == HAS_NONE) {
1,576,600,104✔
2462
    return tRowNoneUpsertColData(aColData, nColData, flag);
7,913✔
2463
  } else if (pRow->flag == HAS_NULL) {
1,576,592,191✔
2464
    return tRowNullUpsertColData(aColData, nColData, pTSchema, flag);
167,565✔
2465
  } else if (pRow->flag >> 4) {  // KV row
1,576,424,626✔
2466
    return tRowKVUpsertColData(pRow, pTSchema, aColData, nColData, flag);
4,286,996✔
2467
  } else {  // TUPLE row
2468
    return tRowTupleUpsertColData(pRow, pTSchema, aColData, nColData, flag);
1,572,137,630✔
2469
  }
2470
}
2471

2472
void tRowGetPrimaryKey(SRow *row, SRowKey *key) {
2,147,483,647✔
2473
  key->numOfPKs = row->numOfPKs;
2,147,483,647✔
2474

2475
  if (key->numOfPKs == 0) {
2,147,483,647!
2476
    return;
×
2477
  }
2478

2479
  SPrimaryKeyIndex indices[TD_MAX_PK_COLS];
2480

2481
  uint8_t *data = row->data;
2,147,483,647✔
2482

2483
  for (int32_t i = 0; i < row->numOfPKs; i++) {
2,147,483,647✔
2484
    data += tGetPrimaryKeyIndex(data, &indices[i]);
2,147,483,647✔
2485
  }
2486

2487
  // primary keys
2488
  for (int32_t i = 0; i < row->numOfPKs; i++) {
2,147,483,647✔
2489
    key->pks[i].type = indices[i].type;
2,147,483,647✔
2490

2491
    uint8_t *tdata = data + indices[i].offset;
2,147,483,647✔
2492
    if (row->flag >> 4) {
2,147,483,647✔
2493
      tdata += tGetI16v(tdata, NULL);
9,509,635✔
2494
    }
2495

2496
    if (IS_VAR_DATA_TYPE(indices[i].type)) {
2,147,483,647!
2497
      key->pks[i].pData = tdata;
956,356✔
2498
      key->pks[i].pData += tGetU32v(key->pks[i].pData, &key->pks[i].nData);
1,912,712!
2499
    } else {
2500
      valueSetDatum(key->pks + i, indices[i].type, tdata, tDataTypes[indices[i].type].bytes);
2,147,483,647✔
2501
    }
2502
  }
2503
}
2504

2505
#define T_COMPARE_SCALAR_VALUE(TYPE, V1, V2)    \
2506
  do {                                          \
2507
    if (*(TYPE *)(V1) < *(TYPE *)(V2)) {        \
2508
      return -1;                                \
2509
    } else if (*(TYPE *)(V1) > *(TYPE *)(V2)) { \
2510
      return 1;                                 \
2511
    } else {                                    \
2512
      return 0;                                 \
2513
    }                                           \
2514
  } while (0)
2515

2516
int32_t tValueCompare(const SValue *tv1, const SValue *tv2) {
92,849,980✔
2517
  switch (tv1->type) {
92,849,980!
2518
    case TSDB_DATA_TYPE_BOOL:
×
2519
    case TSDB_DATA_TYPE_TINYINT:
2520
      T_COMPARE_SCALAR_VALUE(int8_t, &VALUE_GET_TRIVIAL_DATUM(tv1), &VALUE_GET_TRIVIAL_DATUM(tv2));
×
2521
    case TSDB_DATA_TYPE_SMALLINT:
×
2522
      T_COMPARE_SCALAR_VALUE(int16_t, &VALUE_GET_TRIVIAL_DATUM(tv1), &VALUE_GET_TRIVIAL_DATUM(tv2));
×
2523
    case TSDB_DATA_TYPE_INT:
733,665✔
2524
      T_COMPARE_SCALAR_VALUE(int32_t, &VALUE_GET_TRIVIAL_DATUM(tv1), &VALUE_GET_TRIVIAL_DATUM(tv2));
733,665✔
2525
    case TSDB_DATA_TYPE_BIGINT:
90,660,705✔
2526
    case TSDB_DATA_TYPE_TIMESTAMP:
2527
      T_COMPARE_SCALAR_VALUE(int64_t, &VALUE_GET_TRIVIAL_DATUM(tv1), &VALUE_GET_TRIVIAL_DATUM(tv2));
90,660,705!
2528
    case TSDB_DATA_TYPE_FLOAT:
×
2529
      T_COMPARE_SCALAR_VALUE(float, &VALUE_GET_TRIVIAL_DATUM(tv1), &VALUE_GET_TRIVIAL_DATUM(tv2));
×
2530
    case TSDB_DATA_TYPE_DOUBLE:
×
2531
      T_COMPARE_SCALAR_VALUE(double, &VALUE_GET_TRIVIAL_DATUM(tv1), &VALUE_GET_TRIVIAL_DATUM(tv2));
×
2532
    case TSDB_DATA_TYPE_UTINYINT:
×
2533
      T_COMPARE_SCALAR_VALUE(uint8_t, &VALUE_GET_TRIVIAL_DATUM(tv1), &VALUE_GET_TRIVIAL_DATUM(tv2));
×
2534
    case TSDB_DATA_TYPE_USMALLINT:
×
2535
      T_COMPARE_SCALAR_VALUE(uint16_t, &VALUE_GET_TRIVIAL_DATUM(tv1), &VALUE_GET_TRIVIAL_DATUM(tv2));
×
2536
    case TSDB_DATA_TYPE_UINT:
369,730✔
2537
      T_COMPARE_SCALAR_VALUE(uint32_t, &VALUE_GET_TRIVIAL_DATUM(tv1), &VALUE_GET_TRIVIAL_DATUM(tv2));
369,730✔
2538
    case TSDB_DATA_TYPE_UBIGINT:
377,040✔
2539
      T_COMPARE_SCALAR_VALUE(uint64_t, &VALUE_GET_TRIVIAL_DATUM(tv1), &VALUE_GET_TRIVIAL_DATUM(tv2));
377,040✔
2540
    case TSDB_DATA_TYPE_GEOMETRY:
723,921✔
2541
    case TSDB_DATA_TYPE_BINARY: {
2542
      int32_t ret = strncmp((const char *)tv1->pData, (const char *)tv2->pData, TMIN(tv1->nData, tv2->nData));
723,921✔
2543
      return ret ? ret : (tv1->nData < tv2->nData ? -1 : (tv1->nData > tv2->nData ? 1 : 0));
723,921✔
2544
    }
2545
    case TSDB_DATA_TYPE_NCHAR: {
×
2546
      int32_t ret = taosUcs4Compare((TdUcs4 *)tv1->pData, (TdUcs4 *)tv2->pData,
×
2547
                                    tv1->nData < tv2->nData ? tv1->nData : tv2->nData);
×
2548
      return ret ? ret : (tv1->nData < tv2->nData ? -1 : (tv1->nData > tv2->nData ? 1 : 0));
×
2549
    }
2550
    case TSDB_DATA_TYPE_VARBINARY: {
×
2551
      int32_t ret = memcmp(tv1->pData, tv2->pData, tv1->nData < tv2->nData ? tv1->nData : tv2->nData);
×
2552
      return ret ? ret : (tv1->nData < tv2->nData ? -1 : (tv1->nData > tv2->nData ? 1 : 0));
×
2553
    }
2554
    default:
×
2555
      break;
×
2556
  }
2557

2558
  return 0;
×
2559
}
2560

2561
// NOTE:
2562
// set key->numOfPKs to 0 as the smallest key with ts
2563
// set key->numOfPKs to (TD_MAX_PK_COLS + 1) as the largest key with ts
2564
FORCE_INLINE int32_t tRowKeyCompare(const SRowKey *key1, const SRowKey *key2) {
2,147,483,647✔
2565
  if (key1->ts < key2->ts) {
2,147,483,647!
2566
    return -1;
2,147,483,647✔
2567
  } else if (key1->ts > key2->ts) {
2,147,483,647!
2568
    return 1;
2,147,483,647✔
2569
  }
2570

2571
  if (key1->numOfPKs == key2->numOfPKs) {
74,640,909!
2572
    for (uint8_t iKey = 0; iKey < key1->numOfPKs; iKey++) {
114,361,393!
2573
      int32_t ret = tValueCompare(&key1->pks[iKey], &key2->pks[iKey]);
92,859,224✔
2574
      if (ret) return ret;
92,832,683!
2575
    }
2576
  } else if (key1->numOfPKs < key2->numOfPKs) {
×
2577
    return -1;
9✔
2578
  } else {
2579
    return 1;
×
2580
  }
2581

2582
  return 0;
21,502,169✔
2583
}
2584

2585
void tRowKeyAssign(SRowKey *pDst, SRowKey *pSrc) {
1,103,849,533✔
2586
  pDst->ts = pSrc->ts;
1,103,849,533✔
2587
  pDst->numOfPKs = pSrc->numOfPKs;
1,103,849,533✔
2588

2589
  if (pSrc->numOfPKs > 0) {
1,103,849,533✔
2590
    for (int32_t i = 0; i < pSrc->numOfPKs; ++i) {
110,425,382✔
2591
      SValue *pVal = &pDst->pks[i];
54,826,449✔
2592
      pVal->type = pSrc->pks[i].type;
54,826,449✔
2593

2594
      valueCloneDatum(pVal, pSrc->pks + i, pVal->type);
54,826,449✔
2595
    }
2596
  }
2597
}
1,104,623,096✔
2598

2599
// STag ========================================
2600
static int tTagValCmprFn(const void *p1, const void *p2) {
229,042,619✔
2601
  if (((STagVal *)p1)->cid < ((STagVal *)p2)->cid) {
229,042,619✔
2602
    return -1;
78,689,364✔
2603
  } else if (((STagVal *)p1)->cid > ((STagVal *)p2)->cid) {
150,353,255✔
2604
    return 1;
83,062,929✔
2605
  }
2606

2607
  return 0;
67,290,326✔
2608
}
2609
static int tTagValJsonCmprFn(const void *p1, const void *p2) {
23,100✔
2610
  return strcmp(((STagVal *)p1)[0].pKey, ((STagVal *)p2)[0].pKey);
23,100✔
2611
}
2612

2613
#ifdef TD_DEBUG_PRINT_TAG
2614
static void debugPrintTagVal(int8_t type, const void *val, int32_t vlen, const char *tag, int32_t ln) {
2615
  switch (type) {
2616
    case TSDB_DATA_TYPE_VARBINARY:
2617
    case TSDB_DATA_TYPE_JSON:
2618
    case TSDB_DATA_TYPE_VARCHAR:
2619
    case TSDB_DATA_TYPE_NCHAR:
2620
    case TSDB_DATA_TYPE_GEOMETRY: {
2621
      char tmpVal[32] = {0};
2622
      tstrncpy(tmpVal, val, vlen > 31 ? 31 : vlen);
2623
      printf("%s:%d type:%d vlen:%d, val:\"%s\"\n", tag, ln, (int32_t)type, vlen, tmpVal);
2624
    } break;
2625
    case TSDB_DATA_TYPE_FLOAT:
2626
      printf("%s:%d type:%d vlen:%d, val:%f\n", tag, ln, (int32_t)type, vlen, *(float *)val);
2627
      break;
2628
    case TSDB_DATA_TYPE_DOUBLE:
2629
      printf("%s:%d type:%d vlen:%d, val:%lf\n", tag, ln, (int32_t)type, vlen, *(double *)val);
2630
      break;
2631
    case TSDB_DATA_TYPE_BOOL:
2632
      printf("%s:%d type:%d vlen:%d, val:%" PRIu8 "\n", tag, ln, (int32_t)type, vlen, *(uint8_t *)val);
2633
      break;
2634
    case TSDB_DATA_TYPE_TINYINT:
2635
      printf("%s:%d type:%d vlen:%d, val:%" PRIi8 "\n", tag, ln, (int32_t)type, vlen, *(int8_t *)val);
2636
      break;
2637
    case TSDB_DATA_TYPE_SMALLINT:
2638
      printf("%s:%d type:%d vlen:%d, val:%" PRIi16 "\n", tag, ln, (int32_t)type, vlen, *(int16_t *)val);
2639
      break;
2640
    case TSDB_DATA_TYPE_INT:
2641
      printf("%s:%d type:%d vlen:%d, val:%" PRIi32 "\n", tag, ln, (int32_t)type, vlen, *(int32_t *)val);
2642
      break;
2643
    case TSDB_DATA_TYPE_BIGINT:
2644
      printf("%s:%d type:%d vlen:%d, val:%" PRIi64 "\n", tag, ln, (int32_t)type, vlen, *(int64_t *)val);
2645
      break;
2646
    case TSDB_DATA_TYPE_TIMESTAMP:
2647
      printf("%s:%d type:%d vlen:%d, val:%" PRIi64 "\n", tag, ln, (int32_t)type, vlen, *(int64_t *)val);
2648
      break;
2649
    case TSDB_DATA_TYPE_UTINYINT:
2650
      printf("%s:%d type:%d vlen:%d, val:%" PRIu8 "\n", tag, ln, (int32_t)type, vlen, *(uint8_t *)val);
2651
      break;
2652
    case TSDB_DATA_TYPE_USMALLINT:
2653
      printf("%s:%d type:%d vlen:%d, val:%" PRIu16 "\n", tag, ln, (int32_t)type, vlen, *(uint16_t *)val);
2654
      break;
2655
    case TSDB_DATA_TYPE_UINT:
2656
      printf("%s:%d type:%d vlen:%d, val:%" PRIu32 "\n", tag, ln, (int32_t)type, vlen, *(uint32_t *)val);
2657
      break;
2658
    case TSDB_DATA_TYPE_UBIGINT:
2659
      printf("%s:%d type:%d vlen:%d, val:%" PRIu64 "\n", tag, ln, (int32_t)type, vlen, *(uint64_t *)val);
2660
      break;
2661
    case TSDB_DATA_TYPE_NULL:
2662
      printf("%s:%d type:%d vlen:%d, val:%" PRIi8 "\n", tag, ln, (int32_t)type, vlen, *(int8_t *)val);
2663
      break;
2664
    default:
2665
      break;
2666
  }
2667
}
2668

2669
void debugPrintSTag(STag *pTag, const char *tag, int32_t ln) {
2670
  int8_t   isJson = pTag->flags & TD_TAG_JSON;
2671
  int8_t   isLarge = pTag->flags & TD_TAG_LARGE;
2672
  uint8_t *p = NULL;
2673
  int16_t  offset = 0;
2674

2675
  if (isLarge) {
2676
    p = (uint8_t *)&((int16_t *)pTag->idx)[pTag->nTag];
2677
  } else {
2678
    p = (uint8_t *)&pTag->idx[pTag->nTag];
2679
  }
2680
  printf("%s:%d >>> STAG === %s:%s, len: %d, nTag: %d, sver:%d\n", tag, ln, isJson ? "json" : "normal",
2681
         isLarge ? "large" : "small", (int32_t)pTag->len, (int32_t)pTag->nTag, pTag->ver);
2682
  for (uint16_t n = 0; n < pTag->nTag; ++n) {
2683
    if (isLarge) {
2684
      offset = ((int16_t *)pTag->idx)[n];
2685
    } else {
2686
      offset = pTag->idx[n];
2687
    }
2688
    STagVal tagVal = {0};
2689
    if (isJson) {
2690
      tagVal.pKey = (char *)POINTER_SHIFT(p, offset);
2691
    } else {
2692
      tagVal.cid = *(int16_t *)POINTER_SHIFT(p, offset);
2693
    }
2694
    printf("%s:%d loop[%d-%d] offset=%d\n", __func__, __LINE__, (int32_t)pTag->nTag, (int32_t)n, (int32_t)offset);
2695
    tGetTagVal(p + offset, &tagVal, isJson);
2696
    if (IS_VAR_DATA_TYPE(tagVal.type)) {
2697
      debugPrintTagVal(tagVal.type, tagVal.pData, tagVal.nData, __func__, __LINE__);
2698
    } else {
2699
      debugPrintTagVal(tagVal.type, &tagVal.i64, tDataTypes[tagVal.type].bytes, __func__, __LINE__);
2700
    }
2701
  }
2702
  printf("\n");
2703
}
2704
#endif
2705

2706
static int32_t tPutTagVal(uint8_t *p, STagVal *pTagVal, int8_t isJson) {
434,648✔
2707
  int32_t n = 0;
434,648✔
2708

2709
  // key
2710
  if (isJson) {
434,648✔
2711
    n += tPutCStr(p ? p + n : p, pTagVal->pKey);
2,440✔
2712
  } else {
2713
    n += tPutI16v(p ? p + n : p, pTagVal->cid);
866,856✔
2714
  }
2715

2716
  // type
2717
  n += tPutI8(p ? p + n : p, pTagVal->type);
434,648✔
2718

2719
  // value
2720
  if (IS_VAR_DATA_TYPE(pTagVal->type)) {
434,648!
2721
    n += tPutBinary(p ? p + n : p, pTagVal->pData, pTagVal->nData);
296,858✔
2722
  } else {
2723
    p = p ? p + n : p;
286,219✔
2724
    n += tDataTypes[pTagVal->type].bytes;
286,219✔
2725
    if (p) (void)memcpy(p, &(pTagVal->i64), tDataTypes[pTagVal->type].bytes);
286,219✔
2726
  }
2727

2728
  return n;
434,648✔
2729
}
2730
static int32_t tGetTagVal(uint8_t *p, STagVal *pTagVal, int8_t isJson) {
225,745,570✔
2731
  int32_t n = 0;
225,745,570✔
2732

2733
  // key
2734
  if (isJson) {
225,745,570✔
2735
    n += tGetCStr(p + n, &pTagVal->pKey);
48,486!
2736
  } else {
2737
    n += tGetI16v(p + n, &pTagVal->cid);
451,442,654!
2738
  }
2739

2740
  // type
2741
  n += tGetI8(p + n, &pTagVal->type);
225,745,570!
2742

2743
  // value
2744
  if (IS_VAR_DATA_TYPE(pTagVal->type)) {
225,745,570!
2745
    n += tGetBinary(p + n, &pTagVal->pData, &pTagVal->nData);
77,817,116!
2746
  } else {
2747
    (void)memcpy(&(pTagVal->i64), p + n, tDataTypes[pTagVal->type].bytes);
186,837,012✔
2748
    n += tDataTypes[pTagVal->type].bytes;
186,837,012✔
2749
  }
2750

2751
  return n;
225,745,570✔
2752
}
2753

2754
bool tTagIsJson(const void *pTag) { return (((const STag *)pTag)->flags & TD_TAG_JSON); }
44,767✔
2755

2756
bool tTagIsJsonNull(void *data) {
7,798✔
2757
  STag  *pTag = (STag *)data;
7,798✔
2758
  int8_t isJson = tTagIsJson(pTag);
7,798✔
2759
  if (!isJson) return false;
7,798✔
2760
  return ((STag *)data)->nTag == 0;
3,285✔
2761
}
2762

2763
int32_t tTagNew(SArray *pArray, int32_t version, int8_t isJson, STag **ppTag) {
99,058✔
2764
  int32_t  code = 0;
99,058✔
2765
  uint8_t *p = NULL;
99,058✔
2766
  int16_t  n = 0;
99,058✔
2767
  int16_t  nTag = taosArrayGetSize(pArray);
99,058✔
2768
  int32_t  szTag = 0;
99,076✔
2769
  int8_t   isLarge = 0;
99,076✔
2770

2771
  // sort
2772
  if (isJson) {
99,076✔
2773
    taosSort(pArray->pData, nTag, sizeof(STagVal), tTagValJsonCmprFn);
302✔
2774
  } else {
2775
    taosSort(pArray->pData, nTag, sizeof(STagVal), tTagValCmprFn);
98,774✔
2776
  }
2777

2778
  // get size
2779
  for (int16_t iTag = 0; iTag < nTag; iTag++) {
316,481✔
2780
    szTag += tPutTagVal(NULL, (STagVal *)taosArrayGet(pArray, iTag), isJson);
217,405✔
2781
  }
2782
  if (szTag <= INT8_MAX) {
99,076✔
2783
    szTag = szTag + sizeof(STag) + sizeof(int8_t) * nTag;
93,368✔
2784
  } else {
2785
    szTag = szTag + sizeof(STag) + sizeof(int16_t) * nTag;
5,708✔
2786
    isLarge = 1;
5,708✔
2787
  }
2788

2789
  // build tag
2790
  (*ppTag) = (STag *)taosMemoryCalloc(szTag, 1);
99,076!
2791
  if ((*ppTag) == NULL) {
99,128!
2792
    code = terrno;
×
2793
    goto _err;
×
2794
  }
2795
  (*ppTag)->flags = 0;
99,128✔
2796
  if (isJson) {
99,128✔
2797
    (*ppTag)->flags |= TD_TAG_JSON;
302✔
2798
  }
2799
  if (isLarge) {
99,128✔
2800
    (*ppTag)->flags |= TD_TAG_LARGE;
5,723✔
2801
  }
2802
  (*ppTag)->len = szTag;
99,128✔
2803
  (*ppTag)->nTag = nTag;
99,128✔
2804
  (*ppTag)->ver = version;
99,128✔
2805

2806
  if (isLarge) {
99,128✔
2807
    p = (uint8_t *)&((int16_t *)(*ppTag)->idx)[nTag];
5,723✔
2808
  } else {
2809
    p = (uint8_t *)&(*ppTag)->idx[nTag];
93,405✔
2810
  }
2811
  n = 0;
99,128✔
2812
  for (int16_t iTag = 0; iTag < nTag; iTag++) {
316,592✔
2813
    if (isLarge) {
217,459✔
2814
      ((int16_t *)(*ppTag)->idx)[iTag] = n;
67,833✔
2815
    } else {
2816
      (*ppTag)->idx[iTag] = n;
149,626✔
2817
    }
2818
    n += tPutTagVal(p + n, (STagVal *)taosArrayGet(pArray, iTag), isJson);
217,459✔
2819
  }
2820
#ifdef TD_DEBUG_PRINT_TAG
2821
  debugPrintSTag(*ppTag, __func__, __LINE__);
2822
#endif
2823

2824
  return code;
99,133✔
2825

2826
_err:
×
2827
  return code;
×
2828
}
2829

2830
void tTagFree(STag *pTag) {
11,076✔
2831
  if (pTag) taosMemoryFree(pTag);
11,076!
2832
}
11,076✔
2833

2834
char *tTagValToData(const STagVal *value, bool isJson) {
18,310,367✔
2835
  if (!value) {
18,310,367!
2836
    return NULL;
×
2837
  }
2838

2839
  char  *data = NULL;
18,310,367✔
2840
  int8_t typeBytes = 0;
18,310,367✔
2841
  if (isJson) {
18,310,367✔
2842
    typeBytes = CHAR_BYTES;
12,400✔
2843
  }
2844

2845
  if (IS_VAR_DATA_TYPE(value->type)) {
18,310,367✔
2846
    data = taosMemoryCalloc(1, typeBytes + VARSTR_HEADER_SIZE + value->nData);
4,390,985!
2847
    if (data == NULL) {
4,392,048!
2848
      return NULL;
×
2849
    }
2850

2851
    if (isJson) {
4,392,048✔
2852
      *data = value->type;
8,099✔
2853
    }
2854

2855
    varDataLen(data + typeBytes) = value->nData;
4,392,048✔
2856
    (void)memcpy(varDataVal(data + typeBytes), value->pData, value->nData);
4,392,048✔
2857
  } else {
2858
    data = ((char *)&(value->i64)) - typeBytes;  // json with type
13,919,382✔
2859
  }
2860

2861
  return data;
18,311,430✔
2862
}
2863

2864
bool tTagGet(const STag *pTag, STagVal *pTagVal) {
66,342,233✔
2865
  if (!pTag || !pTagVal) {
66,342,233!
2866
    return false;
×
2867
  }
2868

2869
  int16_t  lidx = 0;
66,518,356✔
2870
  int16_t  ridx = pTag->nTag - 1;
66,518,356✔
2871
  int16_t  midx;
2872
  uint8_t *p;
2873
  int8_t   isJson = pTag->flags & TD_TAG_JSON;
66,518,356✔
2874
  int8_t   isLarge = pTag->flags & TD_TAG_LARGE;
66,518,356✔
2875
  int16_t  offset;
2876
  STagVal  tv;
2877
  int      c;
2878

2879
  if (isLarge) {
66,518,356✔
2880
    p = (uint8_t *)&((int16_t *)pTag->idx)[pTag->nTag];
62,310,165✔
2881
  } else {
2882
    p = (uint8_t *)&pTag->idx[pTag->nTag];
4,208,191✔
2883
  }
2884

2885
  pTagVal->type = TSDB_DATA_TYPE_NULL;
66,518,356✔
2886
  pTagVal->pData = NULL;
66,518,356✔
2887
  pTagVal->nData = 0;
66,518,356✔
2888
  while (lidx <= ridx) {
227,903,046✔
2889
    midx = (lidx + ridx) / 2;
226,798,599✔
2890
    if (isLarge) {
226,798,599✔
2891
      offset = ((int16_t *)pTag->idx)[midx];
203,058,891✔
2892
    } else {
2893
      offset = pTag->idx[midx];
23,739,708✔
2894
    }
2895

2896
    int32_t nt = tGetTagVal(p + offset, &tv, isJson);
226,798,599✔
2897
    if (isJson) {
229,573,573✔
2898
      c = tTagValJsonCmprFn(pTagVal, &tv);
22,740✔
2899
    } else {
2900
      c = tTagValCmprFn(pTagVal, &tv);
229,550,833✔
2901
    }
2902

2903
    if (c < 0) {
231,266,954✔
2904
      ridx = midx - 1;
78,306,544✔
2905
    } else if (c > 0) {
152,960,410✔
2906
      lidx = midx + 1;
83,078,146✔
2907
    } else {
2908
      (void)memcpy(pTagVal, &tv, sizeof(tv));
69,882,264✔
2909
      return true;
69,882,264✔
2910
    }
2911
  }
2912
  return false;
1,104,447✔
2913
}
2914

2915
int32_t tEncodeTag(SEncoder *pEncoder, const STag *pTag) {
771,975✔
2916
  return tEncodeBinary(pEncoder, (const uint8_t *)pTag, pTag->len);
1,543,950✔
2917
}
2918

2919
int32_t tDecodeTag(SDecoder *pDecoder, STag **ppTag) { return tDecodeBinary(pDecoder, (uint8_t **)ppTag, NULL); }
35,615,235✔
2920

2921
int32_t tTagToValArray(const STag *pTag, SArray **ppArray) {
1,501✔
2922
  int32_t  code = 0;
1,501✔
2923
  uint8_t *p = NULL;
1,501✔
2924
  STagVal  tv = {0};
1,501✔
2925
  int8_t   isLarge = pTag->flags & TD_TAG_LARGE;
1,501✔
2926
  int16_t  offset = 0;
1,501✔
2927

2928
  if (isLarge) {
1,501✔
2929
    p = (uint8_t *)&((int16_t *)pTag->idx)[pTag->nTag];
23✔
2930
  } else {
2931
    p = (uint8_t *)&pTag->idx[pTag->nTag];
1,478✔
2932
  }
2933

2934
  (*ppArray) = taosArrayInit(pTag->nTag + 1, sizeof(STagVal));
1,501✔
2935
  if (*ppArray == NULL) {
1,501!
2936
    code = terrno;
×
2937
    goto _err;
×
2938
  }
2939

2940
  for (int16_t iTag = 0; iTag < pTag->nTag; iTag++) {
3,731✔
2941
    if (isLarge) {
2,230✔
2942
      offset = ((int16_t *)pTag->idx)[iTag];
29✔
2943
    } else {
2944
      offset = pTag->idx[iTag];
2,201✔
2945
    }
2946
    int32_t nt = tGetTagVal(p + offset, &tv, pTag->flags & TD_TAG_JSON);
2,230✔
2947
    if (taosArrayPush(*ppArray, &tv) == NULL) {
4,460!
2948
      code = terrno;
×
2949
      goto _err;
×
2950
    }
2951
  }
2952

2953
  return code;
1,501✔
2954

2955
_err:
×
2956
  return code;
×
2957
}
2958

2959
// STSchema ========================================
2960
STSchema *tBuildTSchema(SSchema *aSchema, int32_t numOfCols, int32_t version) {
24,272,837✔
2961
  STSchema *pTSchema = taosMemoryCalloc(1, sizeof(STSchema) + sizeof(STColumn) * numOfCols);
24,272,837!
2962
  if (pTSchema == NULL) {
24,331,165!
2963
    terrno = TSDB_CODE_OUT_OF_MEMORY;
×
2964
    return NULL;
×
2965
  }
2966

2967
  pTSchema->numOfCols = numOfCols;
24,331,165✔
2968
  pTSchema->version = version;
24,331,165✔
2969

2970
  // timestamp column
2971
  if (!(aSchema[0].type == TSDB_DATA_TYPE_TIMESTAMP)) {
24,331,165!
2972
    terrno = TSDB_CODE_INVALID_PARA;
×
2973
    taosMemoryFree(pTSchema);
×
2974
    return NULL;
×
2975
  }
2976
  if (!(aSchema[0].colId == PRIMARYKEY_TIMESTAMP_COL_ID)) {
24,331,165!
2977
    terrno = TSDB_CODE_INVALID_PARA;
×
2978
    taosMemoryFree(pTSchema);
×
2979
    return NULL;
×
2980
  }
2981
  pTSchema->columns[0].colId = aSchema[0].colId;
24,331,165✔
2982
  pTSchema->columns[0].type = aSchema[0].type;
24,331,165✔
2983
  pTSchema->columns[0].flags = aSchema[0].flags;
24,331,165✔
2984
  pTSchema->columns[0].bytes = TYPE_BYTES[aSchema[0].type];
24,331,165✔
2985
  pTSchema->columns[0].offset = -1;
24,331,165✔
2986

2987
  // other columns
2988
  for (int32_t iCol = 1; iCol < numOfCols; iCol++) {
280,456,485✔
2989
    SSchema  *pSchema = &aSchema[iCol];
256,125,320✔
2990
    STColumn *pTColumn = &pTSchema->columns[iCol];
256,125,320✔
2991

2992
    pTColumn->colId = pSchema->colId;
256,125,320✔
2993
    pTColumn->type = pSchema->type;
256,125,320✔
2994
    pTColumn->flags = pSchema->flags;
256,125,320✔
2995
    pTColumn->offset = pTSchema->flen;
256,125,320✔
2996

2997
    if (IS_VAR_DATA_TYPE(pSchema->type)) {
256,125,320!
2998
      pTColumn->bytes = pSchema->bytes;
36,223,524✔
2999
      pTSchema->tlen += (TYPE_BYTES[pSchema->type] + pSchema->bytes);  // todo: remove
36,223,524✔
3000
    } else {
3001
      pTColumn->bytes = TYPE_BYTES[pSchema->type];
219,901,796✔
3002
      pTSchema->tlen += TYPE_BYTES[pSchema->type];  // todo: remove
219,901,796✔
3003
    }
3004

3005
    pTSchema->flen += TYPE_BYTES[pTColumn->type];
256,125,320✔
3006
  }
3007

3008
#if 1  // todo : remove this
3009
  pTSchema->tlen += (int32_t)TD_BITMAP_BYTES(numOfCols);
24,331,165✔
3010
#endif
3011

3012
  return pTSchema;
24,331,165✔
3013
}
3014

3015
static int32_t tTColumnCompare(const void *p1, const void *p2) {
14,829,269✔
3016
  if (((STColumn *)p1)->colId < ((STColumn *)p2)->colId) {
14,829,269✔
3017
    return -1;
2,265,464✔
3018
  } else if (((STColumn *)p1)->colId > ((STColumn *)p2)->colId) {
12,563,805✔
3019
    return 1;
8,903,263✔
3020
  }
3021

3022
  return 0;
3,660,542✔
3023
}
3024

3025
const STColumn *tTSchemaSearchColumn(const STSchema *pTSchema, int16_t cid) {
3,661,711✔
3026
  STColumn tcol = {
3,661,711✔
3027
      .colId = cid,
3028
  };
3029

3030
  return taosbsearch(&tcol, pTSchema->columns, pTSchema->numOfCols, sizeof(STColumn), tTColumnCompare, TD_EQ);
3,661,711✔
3031
}
3032

3033
// SColData ========================================
3034
void tColDataDestroy(void *ph) {
35,696,330✔
3035
  if (ph) {
35,696,330!
3036
    SColData *pColData = (SColData *)ph;
35,696,582✔
3037

3038
    tFree(pColData->pBitMap);
35,696,582!
3039
    tFree(pColData->aOffset);
35,696,772!
3040
    tFree(pColData->pData);
35,696,792!
3041
  }
3042
}
35,697,766✔
3043

3044
void tColDataInit(SColData *pColData, int16_t cid, int8_t type, int8_t cflag) {
36,306,073✔
3045
  pColData->cid = cid;
36,306,073✔
3046
  pColData->type = type;
36,306,073✔
3047
  pColData->cflag = cflag;
36,306,073✔
3048
  tColDataClear(pColData);
36,306,073✔
3049
}
36,310,381✔
3050

3051
void tColDataClear(SColData *pColData) {
74,824,786✔
3052
  pColData->numOfNone = 0;
74,824,786✔
3053
  pColData->numOfNull = 0;
74,824,786✔
3054
  pColData->numOfValue = 0;
74,824,786✔
3055
  pColData->nVal = 0;
74,824,786✔
3056
  pColData->flag = 0;
74,824,786✔
3057
  pColData->nData = 0;
74,824,786✔
3058
}
74,824,786✔
3059

3060
void tColDataDeepClear(SColData *pColData) {
76,025✔
3061
  pColData->pBitMap = NULL;
76,025✔
3062
  pColData->aOffset = NULL;
76,025✔
3063
  pColData->pData = NULL;
76,025✔
3064

3065
  tColDataClear(pColData);
76,025✔
3066
}
76,044✔
3067

3068
static FORCE_INLINE int32_t tColDataPutValue(SColData *pColData, uint8_t *pData, uint32_t nData) {
3069
  int32_t  code = 0;
2,147,483,647✔
3070
  uint32_t cvtNData = sizeof(uint64_t);
2,147,483,647✔
3071
  if (IS_VAR_DATA_TYPE(pColData->type)) {
2,147,483,647!
3072
    // TODO
3073
    if (IS_STR_DATA_BLOB(pColData->type)) {
128,317,098!
3074
      code = tRealloc((uint8_t **)(&pColData->aOffset), ((int64_t)(pColData->nVal + 1)) << 2);
×
3075
      if (code) goto _exit;
×
3076
      pColData->aOffset[pColData->nVal] = pColData->nData;
×
3077

3078
      if (nData) {
×
3079
        code = tRealloc(&pColData->pData, pColData->nData + BSE_SEQUECE_SIZE);
×
3080
        if (code) goto _exit;
×
3081
        (void)memcpy(pColData->pData + pColData->nData, pData, BSE_SEQUECE_SIZE);
×
3082
        pColData->nData += BSE_SEQUECE_SIZE;
×
3083
      } else {
3084
        // uint64_t zero = 0;
3085
        // (void)memcpy(pColData->pData + pColData->nData, &zero, BSE_SEQUECE_SIZE);
3086
        // pColData->nData += BSE_SEQUECE_SIZE;
3087
      }
3088

3089
    } else {
3090
      code = tRealloc((uint8_t **)(&pColData->aOffset), ((int64_t)(pColData->nVal + 1)) << 2);
146,579,514!
3091
      if (code) goto _exit;
146,769,015!
3092
      pColData->aOffset[pColData->nVal] = pColData->nData;
146,769,015✔
3093

3094
      if (nData) {
146,769,015!
3095
        code = tRealloc(&pColData->pData, pColData->nData + nData);
141,120,280!
3096
        if (code) goto _exit;
141,722,343!
3097
        (void)memcpy(pColData->pData + pColData->nData, pData, nData);
141,722,343✔
3098
        pColData->nData += nData;
141,722,343✔
3099
      }
3100
    }
3101
  } else {
3102
    if (!(pColData->nData == tDataTypes[pColData->type].bytes * pColData->nVal)) {
2,147,483,647!
3103
      return TSDB_CODE_INVALID_PARA;
×
3104
    }
3105
    code = tRealloc(&pColData->pData, pColData->nData + tDataTypes[pColData->type].bytes);
2,147,483,647!
3106
    if (code) goto _exit;
2,147,483,647!
3107
    if (pData) {
2,147,483,647!
3108
      (void)memcpy(pColData->pData + pColData->nData, pData, TYPE_BYTES[pColData->type]);
2,147,483,647✔
3109
    } else {
3110
      memset(pColData->pData + pColData->nData, 0, TYPE_BYTES[pColData->type]);
54,934,481✔
3111
    }
3112
    pColData->nData += tDataTypes[pColData->type].bytes;
2,147,483,647✔
3113
  }
3114
  pColData->nVal++;
2,147,483,647✔
3115

3116
_exit:
2,147,483,647✔
3117
  return code;
2,147,483,647✔
3118
}
3119
static FORCE_INLINE int32_t tColDataAppendValue00(SColData *pColData, uint8_t *pData, uint32_t nData) {
3,564,028✔
3120
  pColData->flag = HAS_VALUE;
3,564,181✔
3121
  pColData->numOfValue++;
3,564,028✔
3122
  return tColDataPutValue(pColData, pData, nData);
3,566,850✔
3123
}
3124
static FORCE_INLINE int32_t tColDataAppendValue01(SColData *pColData, uint8_t *pData, uint32_t nData) {
245,525✔
3125
  pColData->flag = HAS_NONE;
245,525✔
3126
  pColData->numOfNone++;
245,525✔
3127
  pColData->nVal++;
245,525✔
3128
  return 0;
245,525✔
3129
}
3130
static FORCE_INLINE int32_t tColDataAppendValue02(SColData *pColData, uint8_t *pData, uint32_t nData) {
44,117✔
3131
  pColData->flag = HAS_NULL;
44,177✔
3132
  pColData->numOfNull++;
44,177✔
3133
  pColData->nVal++;
44,177✔
3134
  return 0;
44,117✔
3135
}
3136
static FORCE_INLINE int32_t tColDataAppendValue10(SColData *pColData, uint8_t *pData, uint32_t nData) {
2,410✔
3137
  int32_t code = 0;
2,410✔
3138

3139
  int32_t nBit = BIT1_SIZE(pColData->nVal + 1);
2,410✔
3140
  code = tRealloc(&pColData->pBitMap, nBit);
2,410!
3141
  if (code) return code;
2,410!
3142

3143
  memset(pColData->pBitMap, 0, nBit);
2,410✔
3144
  SET_BIT1_EX(pColData->pBitMap, pColData->nVal, 1);
2,410!
3145

3146
  pColData->flag |= HAS_VALUE;
2,410✔
3147
  pColData->numOfValue++;
2,410✔
3148

3149
  if (pColData->nVal) {
2,410!
3150
    if (IS_VAR_DATA_TYPE(pColData->type)) {
2,409!
3151
      if (IS_STR_DATA_BLOB(pColData->type)) {
248!
3152
        int32_t nOffset = sizeof(int32_t) * pColData->nVal;
×
3153
        code = tRealloc((uint8_t **)(&pColData->aOffset), nOffset);
×
3154
        if (code) return code;
×
3155
        memset(pColData->aOffset, 0, nOffset);
×
3156

3157
      } else {
3158
        int32_t nOffset = sizeof(int32_t) * pColData->nVal;
252✔
3159
        code = tRealloc((uint8_t **)(&pColData->aOffset), nOffset);
252!
3160
        if (code) return code;
252!
3161
        memset(pColData->aOffset, 0, nOffset);
252✔
3162
      }
3163
    } else {
3164
      pColData->nData = tDataTypes[pColData->type].bytes * pColData->nVal;
2,161✔
3165
      code = tRealloc(&pColData->pData, pColData->nData);
2,161!
3166
      if (code) return code;
2,161!
3167
      memset(pColData->pData, 0, pColData->nData);
2,161✔
3168
    }
3169
  }
3170

3171
  return tColDataPutValue(pColData, pData, nData);
2,411✔
3172
}
3173
static FORCE_INLINE int32_t tColDataAppendValue11(SColData *pColData, uint8_t *pData, uint32_t nData) {
18,465,471✔
3174
  pColData->nVal++;
18,465,471✔
3175
  pColData->numOfNone++;
18,465,471✔
3176
  return 0;
18,465,471✔
3177
}
3178
static FORCE_INLINE int32_t tColDataAppendValue12(SColData *pColData, uint8_t *pData, uint32_t nData) {
603✔
3179
  int32_t code = 0;
603✔
3180

3181
  int32_t nBit = BIT1_SIZE(pColData->nVal + 1);
603✔
3182
  code = tRealloc(&pColData->pBitMap, nBit);
603!
3183
  if (code) return code;
603!
3184

3185
  memset(pColData->pBitMap, 0, nBit);
603✔
3186
  SET_BIT1_EX(pColData->pBitMap, pColData->nVal, 1);
603!
3187

3188
  pColData->flag |= HAS_NULL;
603✔
3189
  pColData->numOfNull++;
603✔
3190
  pColData->nVal++;
603✔
3191

3192
  return code;
603✔
3193
}
3194
static FORCE_INLINE int32_t tColDataAppendValue20(SColData *pColData, uint8_t *pData, uint32_t nData) {
19,136✔
3195
  int32_t code = 0;
19,135✔
3196

3197
  int32_t nBit = BIT1_SIZE(pColData->nVal + 1);
19,135✔
3198
  code = tRealloc(&pColData->pBitMap, nBit);
19,136!
3199
  if (code) return code;
19,137!
3200

3201
  memset(pColData->pBitMap, 0, nBit);
19,137✔
3202
  SET_BIT1_EX(pColData->pBitMap, pColData->nVal, 1);
19,137!
3203

3204
  pColData->flag |= HAS_VALUE;
19,137✔
3205
  pColData->numOfValue++;
19,137✔
3206

3207
  if (pColData->nVal) {
19,137!
3208
    if (IS_VAR_DATA_TYPE(pColData->type)) {
19,924!
3209
      int32_t nOffset = sizeof(int32_t) * pColData->nVal;
787✔
3210
      code = tRealloc((uint8_t **)(&pColData->aOffset), nOffset);
787!
3211
      if (code) return code;
787!
3212
      if (!IS_STR_DATA_BLOB(pColData->type)) {
787!
3213
        memset(pColData->aOffset, 0, nOffset);
787✔
3214
      }
3215
    } else {
3216
      pColData->nData = tDataTypes[pColData->type].bytes * pColData->nVal;
18,350✔
3217
      code = tRealloc(&pColData->pData, pColData->nData);
18,350!
3218
      if (code) return code;
18,350!
3219
      memset(pColData->pData, 0, pColData->nData);
18,350✔
3220
    }
3221
  }
3222

3223
  return tColDataPutValue(pColData, pData, nData);
19,137✔
3224
}
3225
static FORCE_INLINE int32_t tColDataAppendValue21(SColData *pColData, uint8_t *pData, uint32_t nData) {
10,080✔
3226
  int32_t code = 0;
10,080✔
3227

3228
  int32_t nBit = BIT1_SIZE(pColData->nVal + 1);
10,080✔
3229
  code = tRealloc(&pColData->pBitMap, nBit);
10,080✔
3230
  if (code) return code;
10,080!
3231

3232
  memset(pColData->pBitMap, 255, nBit);
10,080✔
3233
  SET_BIT1_EX(pColData->pBitMap, pColData->nVal, 0);
10,080!
3234

3235
  pColData->flag |= HAS_NONE;
10,080✔
3236
  pColData->numOfNone++;
10,080✔
3237
  pColData->nVal++;
10,080✔
3238

3239
  return code;
10,080✔
3240
}
3241
static FORCE_INLINE int32_t tColDataAppendValue22(SColData *pColData, uint8_t *pData, uint32_t nData) {
3,569,492✔
3242
  pColData->nVal++;
3,569,492✔
3243
  pColData->numOfNull++;
3,569,492✔
3244
  return 0;
3,569,492✔
3245
}
3246
static FORCE_INLINE int32_t tColDataAppendValue30(SColData *pColData, uint8_t *pData, uint32_t nData) {
5✔
3247
  int32_t code = 0;
5✔
3248

3249
  pColData->flag |= HAS_VALUE;
5✔
3250
  pColData->numOfValue++;
5✔
3251

3252
  uint8_t *pBitMap = NULL;
5✔
3253
  code = tRealloc(&pBitMap, BIT2_SIZE(pColData->nVal + 1));
5!
3254
  if (code) return code;
5!
3255

3256
  for (int32_t iVal = 0; iVal < pColData->nVal; iVal++) {
1,005!
3257
    SET_BIT2_EX(pBitMap, iVal, GET_BIT1(pColData->pBitMap, iVal));
1,000!
3258
  }
3259
  SET_BIT2_EX(pBitMap, pColData->nVal, 2);
5!
3260

3261
  tFree(pColData->pBitMap);
5!
3262
  pColData->pBitMap = pBitMap;
5✔
3263

3264
  if (pColData->nVal) {
5!
3265
    if (IS_VAR_DATA_TYPE(pColData->type)) {
5!
3266
      int32_t nOffset = sizeof(int32_t) * pColData->nVal;
×
3267
      code = tRealloc((uint8_t **)(&pColData->aOffset), nOffset);
×
3268
      if (code) return code;
×
3269
      memset(pColData->aOffset, 0, nOffset);
×
3270
    } else {
3271
      pColData->nData = tDataTypes[pColData->type].bytes * pColData->nVal;
5✔
3272
      code = tRealloc(&pColData->pData, pColData->nData);
5!
3273
      if (code) return code;
5!
3274
      memset(pColData->pData, 0, pColData->nData);
5✔
3275
    }
3276
  }
3277

3278
  return tColDataPutValue(pColData, pData, nData);
5✔
3279
}
3280
static FORCE_INLINE int32_t tColDataAppendValue31(SColData *pColData, uint8_t *pData, uint32_t nData) {
1,989,900✔
3281
  int32_t code = 0;
1,989,900✔
3282

3283
  code = tRealloc(&pColData->pBitMap, BIT1_SIZE(pColData->nVal + 1));
1,989,900!
3284
  if (code) return code;
1,989,900!
3285

3286
  SET_BIT1_EX(pColData->pBitMap, pColData->nVal, 0);
1,989,900✔
3287
  pColData->numOfNone++;
1,989,900✔
3288
  pColData->nVal++;
1,989,900✔
3289

3290
  return code;
1,989,900✔
3291
}
3292
static FORCE_INLINE int32_t tColDataAppendValue32(SColData *pColData, uint8_t *pData, uint32_t nData) {
9,374✔
3293
  int32_t code = 0;
9,374✔
3294

3295
  code = tRealloc(&pColData->pBitMap, BIT1_SIZE(pColData->nVal + 1));
9,374!
3296
  if (code) return code;
9,376!
3297

3298
  SET_BIT1_EX(pColData->pBitMap, pColData->nVal, 1);
9,376!
3299
  pColData->numOfNull++;
9,376✔
3300
  pColData->nVal++;
9,376✔
3301

3302
  return code;
9,376✔
3303
}
3304
static FORCE_INLINE int32_t tColDataAppendValue40(SColData *pColData, uint8_t *pData, uint32_t nData) {
2,147,483,647✔
3305
  pColData->numOfValue++;
2,147,483,647✔
3306
  return tColDataPutValue(pColData, pData, nData);
2,147,483,647✔
3307
}
3308
static FORCE_INLINE int32_t tColDataAppendValue41(SColData *pColData, uint8_t *pData, uint32_t nData) {
119,519✔
3309
  int32_t code = 0;
119,519✔
3310

3311
  pColData->flag |= HAS_NONE;
119,519✔
3312
  pColData->numOfNone++;
119,519✔
3313

3314
  int32_t nBit = BIT1_SIZE(pColData->nVal + 1);
119,519✔
3315
  code = tRealloc(&pColData->pBitMap, nBit);
119,519✔
3316
  if (code) return code;
119,562!
3317

3318
  memset(pColData->pBitMap, 255, nBit);
119,562✔
3319
  SET_BIT1_EX(pColData->pBitMap, pColData->nVal, 0);
119,562✔
3320

3321
  return tColDataPutValue(pColData, NULL, 0);
119,549✔
3322
}
3323
static FORCE_INLINE int32_t tColDataAppendValue42(SColData *pColData, uint8_t *pData, uint32_t nData) {
219,998✔
3324
  int32_t code = 0;
225,300✔
3325

3326
  pColData->flag |= HAS_NULL;
225,300✔
3327
  pColData->numOfNull++;
225,300✔
3328

3329
  int32_t nBit = BIT1_SIZE(pColData->nVal + 1);
225,300✔
3330
  code = tRealloc(&pColData->pBitMap, nBit);
224,996✔
3331
  if (code) return code;
225,302!
3332

3333
  memset(pColData->pBitMap, 255, nBit);
225,302✔
3334
  SET_BIT1_EX(pColData->pBitMap, pColData->nVal, 0);
225,302!
3335

3336
  return tColDataPutValue(pColData, NULL, 0);
225,309✔
3337
}
3338
static FORCE_INLINE int32_t tColDataAppendValue50(SColData *pColData, uint8_t *pData, uint32_t nData) {
1,843,594✔
3339
  int32_t code = 0;
1,845,595✔
3340

3341
  code = tRealloc(&pColData->pBitMap, BIT1_SIZE(pColData->nVal + 1));
1,845,594!
3342
  if (code) return code;
1,845,821!
3343

3344
  SET_BIT1_EX(pColData->pBitMap, pColData->nVal, 1);
1,845,821!
3345
  pColData->numOfValue++;
1,845,821!
3346

3347
  return tColDataPutValue(pColData, pData, nData);
1,848,807✔
3348
}
3349
static FORCE_INLINE int32_t tColDataAppendValue51(SColData *pColData, uint8_t *pData, uint32_t nData) {
12,699,185✔
3350
  int32_t code = 0;
12,699,185✔
3351

3352
  code = tRealloc(&pColData->pBitMap, BIT1_SIZE(pColData->nVal + 1));
12,699,185!
3353
  if (code) return code;
12,699,198!
3354

3355
  SET_BIT1_EX(pColData->pBitMap, pColData->nVal, 0);
12,699,198✔
3356
  pColData->numOfNone++;
12,699,198✔
3357

3358
  return tColDataPutValue(pColData, NULL, 0);
12,700,903✔
3359
}
3360
static FORCE_INLINE int32_t tColDataAppendValue52(SColData *pColData, uint8_t *pData, uint32_t nData) {
9,973✔
3361
  int32_t code = 0;
9,973✔
3362

3363
  pColData->flag |= HAS_NULL;
9,973✔
3364
  pColData->numOfNull++;
9,973✔
3365

3366
  uint8_t *pBitMap = NULL;
9,973✔
3367
  code = tRealloc(&pBitMap, BIT2_SIZE(pColData->nVal + 1));
9,973!
3368
  if (code) return code;
9,973!
3369

3370
  for (int32_t iVal = 0; iVal < pColData->nVal; iVal++) {
3,013,253!
3371
    SET_BIT2_EX(pBitMap, iVal, GET_BIT1(pColData->pBitMap, iVal) ? 2 : 0);
3,003,280!
3372
  }
3373
  SET_BIT2_EX(pBitMap, pColData->nVal, 1);
9,973!
3374

3375
  tFree(pColData->pBitMap);
9,973!
3376
  pColData->pBitMap = pBitMap;
9,973!
3377

3378
  return tColDataPutValue(pColData, NULL, 0);
9,973✔
3379
}
3380
static FORCE_INLINE int32_t tColDataAppendValue60(SColData *pColData, uint8_t *pData, uint32_t nData) {
554,512,480✔
3381
  int32_t code = 0;
554,515,489✔
3382

3383
  code = tRealloc(&pColData->pBitMap, BIT1_SIZE(pColData->nVal + 1));
554,515,089!
3384
  if (code) return code;
555,290,054!
3385
  SET_BIT1_EX(pColData->pBitMap, pColData->nVal, 1);
555,290,054!
3386
  pColData->numOfValue++;
555,290,054✔
3387

3388
  return tColDataPutValue(pColData, pData, nData);
556,284,435✔
3389
}
3390
static FORCE_INLINE int32_t tColDataAppendValue61(SColData *pColData, uint8_t *pData, uint32_t nData) {
26,434✔
3391
  int32_t code = 0;
26,434✔
3392

3393
  pColData->flag |= HAS_NONE;
26,434✔
3394
  pColData->numOfNone++;
26,434✔
3395

3396
  uint8_t *pBitMap = NULL;
26,434✔
3397
  code = tRealloc(&pBitMap, BIT2_SIZE(pColData->nVal + 1));
26,434!
3398
  if (code) return code;
26,434!
3399

3400
  for (int32_t iVal = 0; iVal < pColData->nVal; iVal++) {
3,401,440✔
3401
    SET_BIT2_EX(pBitMap, iVal, GET_BIT1(pColData->pBitMap, iVal) ? 2 : 1);
3,375,006✔
3402
  }
3403
  SET_BIT2_EX(pBitMap, pColData->nVal, 0);
26,434✔
3404

3405
  tFree(pColData->pBitMap);
26,434!
3406
  pColData->pBitMap = pBitMap;
26,435✔
3407

3408
  return tColDataPutValue(pColData, NULL, 0);
26,433✔
3409
}
3410
static FORCE_INLINE int32_t tColDataAppendValue62(SColData *pColData, uint8_t *pData, uint32_t nData) {
33,584,411✔
3411
  int32_t code = 0;
33,598,106✔
3412

3413
  code = tRealloc(&pColData->pBitMap, BIT1_SIZE(pColData->nVal + 1));
33,594,511!
3414
  if (code) return code;
33,602,173!
3415
  SET_BIT1_EX(pColData->pBitMap, pColData->nVal, 0);
33,602,173✔
3416
  pColData->numOfNull++;
33,602,173✔
3417

3418
  return tColDataPutValue(pColData, NULL, 0);
33,614,998✔
3419
}
3420
static FORCE_INLINE int32_t tColDataAppendValue70(SColData *pColData, uint8_t *pData, uint32_t nData) {
528,238✔
3421
  int32_t code = 0;
528,238✔
3422

3423
  code = tRealloc(&pColData->pBitMap, BIT2_SIZE(pColData->nVal + 1));
528,238!
3424
  if (code) return code;
528,238!
3425
  SET_BIT2_EX(pColData->pBitMap, pColData->nVal, 2);
528,238!
3426
  pColData->numOfValue++;
528,238!
3427

3428
  return tColDataPutValue(pColData, pData, nData);
528,238✔
3429
}
3430
static FORCE_INLINE int32_t tColDataAppendValue71(SColData *pColData, uint8_t *pData, uint32_t nData) {
4,459,567✔
3431
  int32_t code = 0;
4,459,567✔
3432

3433
  code = tRealloc(&pColData->pBitMap, BIT2_SIZE(pColData->nVal + 1));
4,459,567!
3434
  if (code) return code;
4,459,567!
3435
  SET_BIT2_EX(pColData->pBitMap, pColData->nVal, 0);
4,459,567✔
3436
  pColData->numOfNone++;
4,459,567!
3437

3438
  return tColDataPutValue(pColData, NULL, 0);
4,459,567✔
3439
}
3440
static FORCE_INLINE int32_t tColDataAppendValue72(SColData *pColData, uint8_t *pData, uint32_t nData) {
450,507✔
3441
  int32_t code = 0;
450,507✔
3442

3443
  code = tRealloc(&pColData->pBitMap, BIT2_SIZE(pColData->nVal + 1));
450,507!
3444
  if (code) return code;
450,507!
3445
  SET_BIT2_EX(pColData->pBitMap, pColData->nVal, 1);
450,507!
3446
  pColData->numOfNull++;
450,507!
3447

3448
  return tColDataPutValue(pColData, NULL, 0);
450,507✔
3449
}
3450
static int32_t (*tColDataAppendValueImpl[8][3])(SColData *pColData, uint8_t *pData, uint32_t nData) = {
3451
    {tColDataAppendValue00, tColDataAppendValue01, tColDataAppendValue02},  // 0
3452
    {tColDataAppendValue10, tColDataAppendValue11, tColDataAppendValue12},  // HAS_NONE
3453
    {tColDataAppendValue20, tColDataAppendValue21, tColDataAppendValue22},  // HAS_NULL
3454
    {tColDataAppendValue30, tColDataAppendValue31, tColDataAppendValue32},  // HAS_NULL|HAS_NONE
3455
    {tColDataAppendValue40, tColDataAppendValue41, tColDataAppendValue42},  // HAS_VALUE
3456
    {tColDataAppendValue50, tColDataAppendValue51, tColDataAppendValue52},  // HAS_VALUE|HAS_NONE
3457
    {tColDataAppendValue60, tColDataAppendValue61, tColDataAppendValue62},  // HAS_VALUE|HAS_NULL
3458
    {tColDataAppendValue70, tColDataAppendValue71, tColDataAppendValue72},  // HAS_VALUE|HAS_NULL|HAS_NONE
3459

3460
    //       VALUE                  NONE                     NULL
3461
};
3462

3463
static FORCE_INLINE int32_t tColDataPutValueBlob(void *pArg, SColData *pColData, uint8_t *pData, uint32_t nData) {
3464
  int32_t  code = 0;
×
3465
  uint8_t  buf[sizeof(uint64_t) + 1] = {0};
×
3466
  uint64_t seq = 0;
×
3467

3468
  int32_t offset = 0;
×
3469
  if (IS_STR_DATA_BLOB(pColData->type)) {
×
3470
    code = tRealloc((uint8_t **)(&pColData->aOffset), ((int64_t)(pColData->nVal + 1)) << 2);
×
3471
    if (code) goto _exit;
×
3472
    pColData->aOffset[pColData->nVal] = pColData->nData;
×
3473
    if (nData) {
×
3474
      SBlobItem item = {.seqOffsetInRow = seq, .len = nData, .data = pData, .type = TSDB_DATA_BLOB_VALUE};
×
3475
      code = tBlobSetPush(pArg, &item, &seq, 0);
×
3476
      if (code != 0) return code;
×
3477

3478
      offset = tPutU64(buf, seq);
×
3479
      code = tRealloc(&pColData->pData, pColData->nData + offset);
×
3480
      if (code != 0) return code;
×
3481
      memcpy(pColData->pData + pColData->nData, buf, offset);
×
3482
      pColData->nData += offset;
×
3483
    } else {
3484
      int8_t type = pData ? TSDB_DATA_BLOB_EMPTY_VALUE : TSDB_DATA_BLOB_NULL_VALUE;
×
3485
      code = addEmptyItemToBlobSet(pArg, type, NULL);
×
3486
    }
3487
  }
3488
  pColData->nVal++;
×
3489

3490
_exit:
×
3491
  return code;
×
3492
}
3493

3494
static FORCE_INLINE int32_t tColDataAppendValueBlob00(void *pArg, SColData *pColData, uint8_t *pData, uint32_t nData) {
×
3495
  pColData->flag = HAS_VALUE;
×
3496
  pColData->numOfValue++;
×
3497
  return tColDataPutValueBlob(pArg, pColData, pData, nData);
×
3498
}
3499
static FORCE_INLINE int32_t tColDataAppendValueBlob01(void *pArg, SColData *pColData, uint8_t *pData, uint32_t nData) {
×
3500
  pColData->flag = HAS_NONE;
×
3501
  pColData->numOfNone++;
×
3502
  pColData->nVal++;
×
3503
  return addEmptyItemToBlobSet(pArg, TSDB_DATA_BLOB_NULL_VALUE, NULL);
×
3504
}
3505
static FORCE_INLINE int32_t tColDataAppendValueBlob02(void *pArg, SColData *pColData, uint8_t *pData, uint32_t nData) {
×
3506
  pColData->flag = HAS_NULL;
×
3507
  pColData->numOfNull++;
×
3508
  pColData->nVal++;
×
3509
  return addEmptyItemToBlobSet(pArg, TSDB_DATA_BLOB_NULL_VALUE, NULL);
×
3510
}
3511
static FORCE_INLINE int32_t tColDataAppendValueBlob10(void *pArg, SColData *pColData, uint8_t *pData, uint32_t nData) {
×
3512
  int32_t code = 0;
×
3513

3514
  int32_t nBit = BIT1_SIZE(pColData->nVal + 1);
×
3515
  code = tRealloc(&pColData->pBitMap, nBit);
×
3516
  if (code) return code;
×
3517

3518
  memset(pColData->pBitMap, 0, nBit);
×
3519
  SET_BIT1_EX(pColData->pBitMap, pColData->nVal, 1);
×
3520

3521
  pColData->flag |= HAS_VALUE;
×
3522
  pColData->numOfValue++;
×
3523

3524
  if (pColData->nVal) {
×
3525
    if (IS_VAR_DATA_TYPE(pColData->type)) {
×
3526
      int32_t nOffset = sizeof(int32_t) * pColData->nVal;
×
3527
      code = tRealloc((uint8_t **)(&pColData->aOffset), nOffset);
×
3528
      if (code) return code;
×
3529
      memset(pColData->aOffset, 0, nOffset);
×
3530
    } else {
3531
      pColData->nData = tDataTypes[pColData->type].bytes * pColData->nVal;
×
3532
      code = tRealloc(&pColData->pData, pColData->nData);
×
3533
      if (code) return code;
×
3534
      memset(pColData->pData, 0, pColData->nData);
×
3535
    }
3536
  }
3537

3538
  return tColDataPutValueBlob(pArg, pColData, pData, nData);
×
3539
}
3540
static FORCE_INLINE int32_t tColDataAppendValueBlob11(void *pArg, SColData *pColData, uint8_t *pData, uint32_t nData) {
×
3541
  pColData->nVal++;
×
3542
  pColData->numOfNone++;
×
3543
  return addEmptyItemToBlobSet(pArg, TSDB_DATA_BLOB_NULL_VALUE, NULL);
×
3544
}
3545
static FORCE_INLINE int32_t tColDataAppendValueBlob12(void *pArg, SColData *pColData, uint8_t *pData, uint32_t nData) {
×
3546
  int32_t code = 0;
×
3547

3548
  int32_t nBit = BIT1_SIZE(pColData->nVal + 1);
×
3549
  code = tRealloc(&pColData->pBitMap, nBit);
×
3550
  if (code) return code;
×
3551

3552
  memset(pColData->pBitMap, 0, nBit);
×
3553
  SET_BIT1_EX(pColData->pBitMap, pColData->nVal, 1);
×
3554

3555
  pColData->flag |= HAS_NULL;
×
3556
  pColData->numOfNull++;
×
3557
  pColData->nVal++;
×
3558
  return addEmptyItemToBlobSet(pArg, TSDB_DATA_BLOB_NULL_VALUE, NULL);
×
3559
}
3560
static FORCE_INLINE int32_t tColDataAppendValueBlob20(void *pArg, SColData *pColData, uint8_t *pData, uint32_t nData) {
×
3561
  int32_t code = 0;
×
3562

3563
  int32_t nBit = BIT1_SIZE(pColData->nVal + 1);
×
3564
  code = tRealloc(&pColData->pBitMap, nBit);
×
3565
  if (code) return code;
×
3566

3567
  memset(pColData->pBitMap, 0, nBit);
×
3568
  SET_BIT1_EX(pColData->pBitMap, pColData->nVal, 1);
×
3569

3570
  pColData->flag |= HAS_VALUE;
×
3571
  pColData->numOfValue++;
×
3572

3573
  if (pColData->nVal) {
×
3574
    if (IS_STR_DATA_BLOB(pColData->type)) {
×
3575
      int32_t nOffset = sizeof(int32_t) * pColData->nVal;
×
3576
      code = tRealloc((uint8_t **)(&pColData->aOffset), nOffset);
×
3577
      if (code) return code;
×
3578
      memset(pColData->aOffset, 0, nOffset);
×
3579
    } else {
3580
      return TSDB_CODE_INVALID_MSG;
×
3581
    }
3582
  }
3583

3584
  return tColDataPutValueBlob(pArg, pColData, pData, nData);
×
3585
}
3586
static FORCE_INLINE int32_t tColDataAppendValueBlob21(void *pArg, SColData *pColData, uint8_t *pData, uint32_t nData) {
×
3587
  int32_t code = 0;
×
3588

3589
  int32_t nBit = BIT1_SIZE(pColData->nVal + 1);
×
3590
  code = tRealloc(&pColData->pBitMap, nBit);
×
3591
  if (code) return code;
×
3592

3593
  memset(pColData->pBitMap, 255, nBit);
×
3594
  SET_BIT1_EX(pColData->pBitMap, pColData->nVal, 0);
×
3595

3596
  pColData->flag |= HAS_NONE;
×
3597
  pColData->numOfNone++;
×
3598
  pColData->nVal++;
×
3599
  return addEmptyItemToBlobSet(pArg, TSDB_DATA_BLOB_NULL_VALUE, NULL);
×
3600
}
3601
static FORCE_INLINE int32_t tColDataAppendValueBlob22(void *pArg, SColData *pColData, uint8_t *pData, uint32_t nData) {
×
3602
  pColData->nVal++;
×
3603
  pColData->numOfNull++;
×
3604
  return addEmptyItemToBlobSet(pArg, TSDB_DATA_BLOB_NULL_VALUE, NULL);
×
3605
}
3606
static FORCE_INLINE int32_t tColDataAppendValueBlob30(void *pArg, SColData *pColData, uint8_t *pData, uint32_t nData) {
×
3607
  int32_t code = 0;
×
3608

3609
  pColData->flag |= HAS_VALUE;
×
3610
  pColData->numOfValue++;
×
3611

3612
  uint8_t *pBitMap = NULL;
×
3613
  code = tRealloc(&pBitMap, BIT2_SIZE(pColData->nVal + 1));
×
3614
  if (code) return code;
×
3615

3616
  for (int32_t iVal = 0; iVal < pColData->nVal; iVal++) {
×
3617
    SET_BIT2_EX(pBitMap, iVal, GET_BIT1(pColData->pBitMap, iVal));
×
3618
  }
3619
  SET_BIT2_EX(pBitMap, pColData->nVal, 2);
×
3620

3621
  tFree(pColData->pBitMap);
×
3622
  pColData->pBitMap = pBitMap;
×
3623

3624
  if (pColData->nVal) {
×
3625
    if (IS_STR_DATA_BLOB(pColData->type)) {
×
3626
      int32_t nOffset = sizeof(int32_t) * pColData->nVal;
×
3627
      code = tRealloc((uint8_t **)(&pColData->aOffset), nOffset);
×
3628
      if (code) return code;
×
3629
      memset(pColData->aOffset, 0, nOffset);
×
3630
    } else {
3631
      return TSDB_CODE_INVALID_MSG;
×
3632
    }
3633
  }
3634

3635
  return tColDataPutValueBlob(pArg, pColData, pData, nData);
×
3636
}
3637
static FORCE_INLINE int32_t tColDataAppendValueBlob31(void *pArg, SColData *pColData, uint8_t *pData, uint32_t nData) {
×
3638
  int32_t code = 0;
×
3639

3640
  code = tRealloc(&pColData->pBitMap, BIT1_SIZE(pColData->nVal + 1));
×
3641
  if (code) return code;
×
3642

3643
  SET_BIT1_EX(pColData->pBitMap, pColData->nVal, 0);
×
3644
  pColData->numOfNone++;
×
3645
  pColData->nVal++;
×
3646

3647
  return addEmptyItemToBlobSet(pArg, TSDB_DATA_BLOB_NULL_VALUE, NULL);
×
3648
}
3649
static FORCE_INLINE int32_t tColDataAppendValueBlob32(void *pArg, SColData *pColData, uint8_t *pData, uint32_t nData) {
×
3650
  int32_t code = 0;
×
3651

3652
  code = tRealloc(&pColData->pBitMap, BIT1_SIZE(pColData->nVal + 1));
×
3653
  if (code) return code;
×
3654

3655
  SET_BIT1_EX(pColData->pBitMap, pColData->nVal, 1);
×
3656
  pColData->numOfNull++;
×
3657
  pColData->nVal++;
×
3658
  return addEmptyItemToBlobSet(pArg, TSDB_DATA_BLOB_NULL_VALUE, NULL);
×
3659
}
3660
static FORCE_INLINE int32_t tColDataAppendValueBlob40(void *pArg, SColData *pColData, uint8_t *pData, uint32_t nData) {
×
3661
  pColData->numOfValue++;
×
3662
  return tColDataPutValueBlob(pArg, pColData, pData, nData);
×
3663
}
3664
static FORCE_INLINE int32_t tColDataAppendValueBlob41(void *pArg, SColData *pColData, uint8_t *pData, uint32_t nData) {
×
3665
  int32_t code = 0;
×
3666

3667
  pColData->flag |= HAS_NONE;
×
3668
  pColData->numOfNone++;
×
3669

3670
  int32_t nBit = BIT1_SIZE(pColData->nVal + 1);
×
3671
  code = tRealloc(&pColData->pBitMap, nBit);
×
3672
  if (code) return code;
×
3673

3674
  memset(pColData->pBitMap, 255, nBit);
×
3675
  SET_BIT1_EX(pColData->pBitMap, pColData->nVal, 0);
×
3676

3677
  return tColDataPutValueBlob(pArg, pColData, NULL, 0);
×
3678
}
3679
static FORCE_INLINE int32_t tColDataAppendValueBlob42(void *pArg, SColData *pColData, uint8_t *pData, uint32_t nData) {
×
3680
  int32_t code = 0;
×
3681

3682
  pColData->flag |= HAS_NULL;
×
3683
  pColData->numOfNull++;
×
3684

3685
  int32_t nBit = BIT1_SIZE(pColData->nVal + 1);
×
3686
  code = tRealloc(&pColData->pBitMap, nBit);
×
3687
  if (code) return code;
×
3688

3689
  memset(pColData->pBitMap, 255, nBit);
×
3690
  SET_BIT1_EX(pColData->pBitMap, pColData->nVal, 0);
×
3691

3692
  return tColDataPutValueBlob(pArg, pColData, NULL, 0);
×
3693
}
3694
static FORCE_INLINE int32_t tColDataAppendValueBlob50(void *pArg, SColData *pColData, uint8_t *pData, uint32_t nData) {
×
3695
  int32_t code = 0;
×
3696

3697
  code = tRealloc(&pColData->pBitMap, BIT1_SIZE(pColData->nVal + 1));
×
3698
  if (code) return code;
×
3699

3700
  SET_BIT1_EX(pColData->pBitMap, pColData->nVal, 1);
×
3701
  pColData->numOfValue++;
×
3702

3703
  return tColDataPutValueBlob(pArg, pColData, pData, nData);
×
3704
}
3705
static FORCE_INLINE int32_t tColDataAppendValueBlob51(void *pArg, SColData *pColData, uint8_t *pData, uint32_t nData) {
×
3706
  int32_t code = 0;
×
3707

3708
  code = tRealloc(&pColData->pBitMap, BIT1_SIZE(pColData->nVal + 1));
×
3709
  if (code) return code;
×
3710

3711
  SET_BIT1_EX(pColData->pBitMap, pColData->nVal, 0);
×
3712
  pColData->numOfNone++;
×
3713

3714
  return tColDataPutValueBlob(pArg, pColData, NULL, 0);
×
3715
}
3716
static FORCE_INLINE int32_t tColDataAppendValueBlob52(void *pArg, SColData *pColData, uint8_t *pData, uint32_t nData) {
×
3717
  int32_t code = 0;
×
3718

3719
  pColData->flag |= HAS_NULL;
×
3720
  pColData->numOfNull++;
×
3721

3722
  uint8_t *pBitMap = NULL;
×
3723
  code = tRealloc(&pBitMap, BIT2_SIZE(pColData->nVal + 1));
×
3724
  if (code) return code;
×
3725

3726
  for (int32_t iVal = 0; iVal < pColData->nVal; iVal++) {
×
3727
    SET_BIT2_EX(pBitMap, iVal, GET_BIT1(pColData->pBitMap, iVal) ? 2 : 0);
×
3728
  }
3729
  SET_BIT2_EX(pBitMap, pColData->nVal, 1);
×
3730

3731
  tFree(pColData->pBitMap);
×
3732
  pColData->pBitMap = pBitMap;
×
3733

3734
  return tColDataPutValueBlob(pArg, pColData, NULL, 0);
×
3735
}
3736
static FORCE_INLINE int32_t tColDataAppendValueBlob60(void *pArg, SColData *pColData, uint8_t *pData, uint32_t nData) {
×
3737
  int32_t code = 0;
×
3738

3739
  code = tRealloc(&pColData->pBitMap, BIT1_SIZE(pColData->nVal + 1));
×
3740
  if (code) return code;
×
3741
  SET_BIT1_EX(pColData->pBitMap, pColData->nVal, 1);
×
3742
  pColData->numOfValue++;
×
3743

3744
  return tColDataPutValueBlob(pArg, pColData, pData, nData);
×
3745
}
3746
static FORCE_INLINE int32_t tColDataAppendValueBlob61(void *pArg, SColData *pColData, uint8_t *pData, uint32_t nData) {
×
3747
  int32_t code = 0;
×
3748

3749
  pColData->flag |= HAS_NONE;
×
3750
  pColData->numOfNone++;
×
3751

3752
  uint8_t *pBitMap = NULL;
×
3753
  code = tRealloc(&pBitMap, BIT2_SIZE(pColData->nVal + 1));
×
3754
  if (code) return code;
×
3755

3756
  for (int32_t iVal = 0; iVal < pColData->nVal; iVal++) {
×
3757
    SET_BIT2_EX(pBitMap, iVal, GET_BIT1(pColData->pBitMap, iVal) ? 2 : 1);
×
3758
  }
3759
  SET_BIT2_EX(pBitMap, pColData->nVal, 0);
×
3760

3761
  tFree(pColData->pBitMap);
×
3762
  pColData->pBitMap = pBitMap;
×
3763

3764
  return tColDataPutValueBlob(pArg, pColData, NULL, 0);
×
3765
}
3766
static FORCE_INLINE int32_t tColDataAppendValueBlob62(void *pArg, SColData *pColData, uint8_t *pData, uint32_t nData) {
×
3767
  int32_t code = 0;
×
3768

3769
  code = tRealloc(&pColData->pBitMap, BIT1_SIZE(pColData->nVal + 1));
×
3770
  if (code) return code;
×
3771
  SET_BIT1_EX(pColData->pBitMap, pColData->nVal, 0);
×
3772
  pColData->numOfNull++;
×
3773

3774
  return tColDataPutValueBlob(pArg, pColData, NULL, 0);
×
3775
}
3776
static FORCE_INLINE int32_t tColDataAppendValueBlob70(void *pArg, SColData *pColData, uint8_t *pData, uint32_t nData) {
×
3777
  int32_t code = 0;
×
3778

3779
  code = tRealloc(&pColData->pBitMap, BIT2_SIZE(pColData->nVal + 1));
×
3780
  if (code) return code;
×
3781
  SET_BIT2_EX(pColData->pBitMap, pColData->nVal, 2);
×
3782
  pColData->numOfValue++;
×
3783

3784
  return tColDataPutValueBlob(pArg, pColData, pData, nData);
×
3785
}
3786
static FORCE_INLINE int32_t tColDataAppendValueBlob71(void *pArg, SColData *pColData, uint8_t *pData, uint32_t nData) {
×
3787
  int32_t code = 0;
×
3788

3789
  code = tRealloc(&pColData->pBitMap, BIT2_SIZE(pColData->nVal + 1));
×
3790
  if (code) return code;
×
3791
  SET_BIT2_EX(pColData->pBitMap, pColData->nVal, 0);
×
3792
  pColData->numOfNone++;
×
3793

3794
  return tColDataPutValueBlob(pArg, pColData, NULL, 0);
×
3795
}
3796
static FORCE_INLINE int32_t tColDataAppendValueBlob72(void *pArg, SColData *pColData, uint8_t *pData, uint32_t nData) {
×
3797
  int32_t code = 0;
×
3798

3799
  code = tRealloc(&pColData->pBitMap, BIT2_SIZE(pColData->nVal + 1));
×
3800
  if (code) return code;
×
3801
  SET_BIT2_EX(pColData->pBitMap, pColData->nVal, 1);
×
3802
  pColData->numOfNull++;
×
3803

3804
  return tColDataPutValueBlob(pArg, pColData, NULL, 0);
×
3805
}
3806

3807
static int32_t (*tColDataAppendValueBlobImpl[8][3])(void *pDst, SColData *pColData, uint8_t *pData, uint32_t nData) = {
3808
    {tColDataAppendValueBlob00, tColDataAppendValueBlob01, tColDataAppendValueBlob02},  // 0
3809
    {tColDataAppendValueBlob10, tColDataAppendValueBlob11, tColDataAppendValueBlob12},  // HAS_NONE
3810
    {tColDataAppendValueBlob20, tColDataAppendValueBlob21, tColDataAppendValueBlob22},  // HAS_NULL
3811
    {tColDataAppendValueBlob30, tColDataAppendValueBlob31, tColDataAppendValueBlob32},  // HAS_NULL|HAS_NONE
3812
    {tColDataAppendValueBlob40, tColDataAppendValueBlob41, tColDataAppendValueBlob42},  // HAS_VALUE
3813
    {tColDataAppendValueBlob50, tColDataAppendValueBlob51, tColDataAppendValueBlob52},  // HAS_VALUE|HAS_NONE
3814
    {tColDataAppendValueBlob60, tColDataAppendValueBlob61, tColDataAppendValueBlob62},  // HAS_VALUE|HAS_NULL
3815
    {tColDataAppendValueBlob70, tColDataAppendValueBlob71, tColDataAppendValueBlob72}   // HAS_VALUE|HAS_NULL|HAS_NONE
3816
    //       VALUE                  NONE                     NULL
3817
};
3818
int32_t tColDataAppendValue(SColData *pColData, SColVal *pColVal) {
2,147,483,647✔
3819
  if (!(pColData->cid == pColVal->cid && pColData->type == pColVal->value.type)) {
2,147,483,647!
3820
    return TSDB_CODE_INVALID_PARA;
×
3821
  }
3822
  return tColDataAppendValueImpl[pColData->flag][pColVal->flag](
2,147,483,647✔
3823
      pColData, VALUE_GET_DATUM(&pColVal->value, pColData->type), pColVal->value.nData);
2,147,483,647!
3824
}
3825

3826
static FORCE_INLINE int32_t tColDataUpdateValue10(SColData *pColData, uint8_t *pData, uint32_t nData, bool forward) {
7✔
3827
  pColData->numOfNone--;
7✔
3828
  pColData->nVal--;
7✔
3829
  if (pColData->numOfNone) {
7!
3830
    return tColDataAppendValue10(pColData, pData, nData);
×
3831
  } else {
3832
    pColData->flag = 0;
7✔
3833
    return tColDataAppendValue00(pColData, pData, nData);
7✔
3834
  }
3835
}
3836
static FORCE_INLINE int32_t tColDataUpdateValue12(SColData *pColData, uint8_t *pData, uint32_t nData, bool forward) {
24✔
3837
  pColData->numOfNone--;
24✔
3838
  pColData->nVal--;
24✔
3839
  if (pColData->numOfNone) {
24!
3840
    return tColDataAppendValue12(pColData, pData, nData);
×
3841
  } else {
3842
    pColData->flag = 0;
24✔
3843
    return tColDataAppendValue02(pColData, pData, nData);
24✔
3844
  }
3845
  return 0;
3846
}
3847
static FORCE_INLINE int32_t tColDataUpdateValue20(SColData *pColData, uint8_t *pData, uint32_t nData, bool forward) {
146✔
3848
  if (forward) {
146!
3849
    pColData->numOfNull--;
146✔
3850
    pColData->nVal--;
146✔
3851
    if (pColData->numOfNull) {
146!
3852
      return tColDataAppendValue20(pColData, pData, nData);
×
3853
    } else {
3854
      pColData->flag = 0;
146✔
3855
      return tColDataAppendValue00(pColData, pData, nData);
146✔
3856
    }
3857
  }
3858
  return 0;
×
3859
}
3860
static FORCE_INLINE int32_t tColDataUpdateValue30(SColData *pColData, uint8_t *pData, uint32_t nData, bool forward) {
×
3861
  if (GET_BIT1(pColData->pBitMap, pColData->nVal - 1) == 0) {  // NONE ==> VALUE
×
3862
    pColData->numOfNone--;
×
3863
    pColData->nVal--;
×
3864
    if (pColData->numOfNone) {
×
3865
      return tColDataAppendValue30(pColData, pData, nData);
×
3866
    } else {
3867
      pColData->flag = HAS_NULL;
×
3868
      return tColDataAppendValue20(pColData, pData, nData);
×
3869
    }
3870
  } else if (forward) {  // NULL ==> VALUE
×
3871
    pColData->numOfNull--;
×
3872
    pColData->nVal--;
×
3873
    if (pColData->numOfNull) {
×
3874
      return tColDataAppendValue30(pColData, pData, nData);
×
3875
    } else {
3876
      pColData->flag = HAS_NONE;
×
3877
      return tColDataAppendValue10(pColData, pData, nData);
×
3878
    }
3879
  }
3880
  return 0;
×
3881
}
3882
static FORCE_INLINE int32_t tColDataUpdateValue32(SColData *pColData, uint8_t *pData, uint32_t nData, bool forward) {
80✔
3883
  if (GET_BIT1(pColData->pBitMap, pColData->nVal - 1) == 0) {  // NONE ==> NULL
80!
3884
    pColData->numOfNone--;
80✔
3885
    pColData->numOfNull++;
80✔
3886
    if (pColData->numOfNone) {
80!
3887
      SET_BIT1(pColData->pBitMap, pColData->nVal - 1, 1);
×
3888
    } else {
3889
      pColData->flag = HAS_NULL;
80✔
3890
    }
3891
  }
3892
  return 0;
80✔
3893
}
3894
static FORCE_INLINE int32_t tColDataUpdateValue40(SColData *pColData, uint8_t *pData, uint32_t nData, bool forward) {
15,742,915✔
3895
  if (forward) {  // VALUE ==> VALUE
15,742,915!
3896
    pColData->nVal--;
15,744,528✔
3897
    if (IS_VAR_DATA_TYPE(pColData->type)) {
15,744,528!
3898
      pColData->nData = pColData->aOffset[pColData->nVal];
2,649,078✔
3899
    } else {
3900
      pColData->nData -= TYPE_BYTES[pColData->type];
13,095,450✔
3901
    }
3902
    return tColDataPutValue(pColData, pData, nData);
15,868,596✔
3903
  }
3904
  return 0;
×
3905
}
3906
static FORCE_INLINE int32_t tColDataUpdateValue42(SColData *pColData, uint8_t *pData, uint32_t nData, bool forward) {
5,034✔
3907
  if (forward) {  // VALUE ==> NULL
5,034!
3908
    pColData->numOfValue--;
5,034✔
3909
    pColData->nVal--;
5,034✔
3910
    if (pColData->numOfValue) {
5,034✔
3911
      if (IS_VAR_DATA_TYPE(pColData->type)) {
4,998!
3912
        pColData->nData = pColData->aOffset[pColData->nVal];
999✔
3913
      } else {
3914
        pColData->nData -= TYPE_BYTES[pColData->type];
3,999✔
3915
      }
3916
      return tColDataAppendValue42(pColData, pData, nData);
4,999✔
3917
    } else {
3918
      pColData->flag = 0;
36✔
3919
      pColData->nData = 0;
36✔
3920
      return tColDataAppendValue02(pColData, pData, nData);
36✔
3921
    }
3922
  }
3923
  return 0;
×
3924
}
3925
static FORCE_INLINE int32_t tColDataUpdateValue50(SColData *pColData, uint8_t *pData, uint32_t nData, bool forward) {
29,021✔
3926
  if (GET_BIT1(pColData->pBitMap, pColData->nVal - 1) == 0) {  // NONE ==> VALUE
29,021✔
3927
    pColData->numOfNone--;
5,593✔
3928
    pColData->nVal--;
5,593✔
3929
    if (!IS_VAR_DATA_TYPE(pColData->type)) {
5,593!
3930
      pColData->nData -= TYPE_BYTES[pColData->type];
4,474✔
3931
    }
3932
    if (pColData->numOfNone) {
5,593✔
3933
      return tColDataAppendValue50(pColData, pData, nData);
2,000✔
3934
    } else {
3935
      pColData->flag = HAS_VALUE;
3,593✔
3936
      return tColDataAppendValue40(pColData, pData, nData);
3,596✔
3937
    }
3938
  } else if (forward) {  // VALUE ==> VALUE
23,428!
3939
    pColData->nVal--;
23,430✔
3940
    if (IS_VAR_DATA_TYPE(pColData->type)) {
23,430!
3941
      pColData->nData = pColData->aOffset[pColData->nVal];
3,351✔
3942
    } else {
3943
      pColData->nData -= TYPE_BYTES[pColData->type];
20,079✔
3944
    }
3945
    return tColDataPutValue(pColData, pData, nData);
23,436✔
3946
  }
3947
  return 0;
×
3948
}
3949
static FORCE_INLINE int32_t tColDataUpdateValue52(SColData *pColData, uint8_t *pData, uint32_t nData, bool forward) {
304✔
3950
  if (GET_BIT1(pColData->pBitMap, pColData->nVal - 1) == 0) {  // NONE ==> NULL
304!
3951
    pColData->numOfNone--;
304✔
3952
    pColData->nVal--;
304✔
3953
    if (!IS_VAR_DATA_TYPE(pColData->type)) {
304!
3954
      pColData->nData -= TYPE_BYTES[pColData->type];
244✔
3955
    }
3956
    if (pColData->numOfNone) {
304!
3957
      return tColDataAppendValue52(pColData, pData, nData);
×
3958
    } else {
3959
      pColData->flag = HAS_VALUE;
304!
3960
      return tColDataAppendValue42(pColData, pData, nData);
304✔
3961
    }
3962
  } else if (forward) {  // VALUE ==> NULL
×
3963
    pColData->numOfValue--;
×
3964
    pColData->nVal--;
×
3965
    if (pColData->numOfValue) {
×
3966
      if (IS_VAR_DATA_TYPE(pColData->type)) {
×
3967
        pColData->nData = pColData->aOffset[pColData->nVal];
×
3968
      } else {
3969
        pColData->nData -= TYPE_BYTES[pColData->type];
×
3970
      }
3971
      return tColDataAppendValue52(pColData, pData, nData);
×
3972
    } else {
3973
      pColData->flag = HAS_NONE;
×
3974
      pColData->nData = 0;
×
3975
      return tColDataAppendValue12(pColData, pData, nData);
×
3976
    }
3977
  }
3978
  return 0;
×
3979
}
3980
static FORCE_INLINE int32_t tColDataUpdateValue60(SColData *pColData, uint8_t *pData, uint32_t nData, bool forward) {
32,350✔
3981
  if (forward) {
32,350!
3982
    if (GET_BIT1(pColData->pBitMap, pColData->nVal - 1) == 0) {  // NULL ==> VALUE
32,350✔
3983
      pColData->numOfNull--;
22,627✔
3984
      pColData->nVal--;
22,627✔
3985
      if (!IS_VAR_DATA_TYPE(pColData->type)) {
22,627!
3986
        pColData->nData -= TYPE_BYTES[pColData->type];
17,945✔
3987
      }
3988
      if (pColData->numOfNull) {
22,627✔
3989
        return tColDataAppendValue60(pColData, pData, nData);
2,609✔
3990
      } else {
3991
        pColData->flag = HAS_VALUE;
20,018✔
3992
        return tColDataAppendValue40(pColData, pData, nData);
20,020✔
3993
      }
3994
    } else {  // VALUE ==> VALUE
3995
      pColData->nVal--;
9,723✔
3996
      if (IS_VAR_DATA_TYPE(pColData->type)) {
9,723!
3997
        pColData->nData = pColData->aOffset[pColData->nVal];
2,356✔
3998
      } else {
3999
        pColData->nData -= TYPE_BYTES[pColData->type];
7,367✔
4000
      }
4001
      return tColDataPutValue(pColData, pData, nData);
9,723✔
4002
    }
4003
  }
4004
  return 0;
×
4005
}
4006
static FORCE_INLINE int32_t tColDataUpdateValue62(SColData *pColData, uint8_t *pData, uint32_t nData, bool forward) {
24,476✔
4007
  if (forward && (GET_BIT1(pColData->pBitMap, pColData->nVal - 1) == 1)) {  // VALUE ==> NULL
24,476!
4008
    pColData->numOfValue--;
10,099✔
4009
    pColData->nVal--;
10,099✔
4010
    if (pColData->numOfValue) {
10,099!
4011
      if (IS_VAR_DATA_TYPE(pColData->type)) {
10,100!
4012
        pColData->nData = pColData->aOffset[pColData->nVal];
2,200✔
4013
      } else {
4014
        pColData->nData -= TYPE_BYTES[pColData->type];
7,900✔
4015
      }
4016
      return tColDataAppendValue62(pColData, pData, nData);
10,095✔
4017
    } else {
4018
      pColData->flag = HAS_NULL;
×
4019
      pColData->nData = 0;
×
4020
      return tColDataAppendValue20(pColData, pData, nData);
×
4021
    }
4022
  }
4023
  return 0;
14,377✔
4024
}
4025
static FORCE_INLINE int32_t tColDataUpdateValue70(SColData *pColData, uint8_t *pData, uint32_t nData, bool forward) {
401✔
4026
  int32_t code = 0;
401✔
4027

4028
  uint8_t bv = GET_BIT2(pColData->pBitMap, pColData->nVal - 1);
401✔
4029
  if (bv == 0) {  // NONE ==> VALUE
401✔
4030
    pColData->numOfNone--;
400✔
4031
    pColData->nVal--;
400✔
4032
    if (!IS_VAR_DATA_TYPE(pColData->type)) {
400!
4033
      pColData->nData -= TYPE_BYTES[pColData->type];
320✔
4034
    }
4035
    if (pColData->numOfNone) {
400!
4036
      return tColDataAppendValue70(pColData, pData, nData);
×
4037
    } else {
4038
      for (int32_t iVal = 0; iVal < pColData->nVal; ++iVal) {
21,200✔
4039
        SET_BIT1(pColData->pBitMap, iVal, GET_BIT2(pColData->pBitMap, iVal) - 1);
20,800✔
4040
      }
4041
      pColData->flag = (HAS_VALUE | HAS_NULL);
400!
4042
      return tColDataAppendValue60(pColData, pData, nData);
400✔
4043
    }
4044
  } else if (bv == 1) {  // NULL ==> VALUE
1!
4045
    if (forward) {
1!
4046
      pColData->numOfNull--;
1✔
4047
      pColData->nVal--;
1✔
4048
      if (!IS_VAR_DATA_TYPE(pColData->type)) {
1!
4049
        pColData->nData -= TYPE_BYTES[pColData->type];
1✔
4050
      }
4051
      if (pColData->numOfNull) {
1!
4052
        return tColDataAppendValue70(pColData, pData, nData);
×
4053
      } else {
4054
        for (int32_t iVal = 0; iVal < pColData->nVal; ++iVal) {
3✔
4055
          SET_BIT1(pColData->pBitMap, iVal, GET_BIT2(pColData->pBitMap, iVal) ? 1 : 0);
2✔
4056
        }
4057
        pColData->flag = (HAS_VALUE | HAS_NONE);
1!
4058
        return tColDataAppendValue50(pColData, pData, nData);
1✔
4059
      }
4060
    }
4061
  } else if (bv == 2) {  // VALUE ==> VALUE
×
4062
    if (forward) {
×
4063
      pColData->nVal--;
×
4064
      if (IS_VAR_DATA_TYPE(pColData->type)) {
×
4065
        pColData->nData = pColData->aOffset[pColData->nVal];
×
4066
      } else {
4067
        pColData->nData -= TYPE_BYTES[pColData->type];
×
4068
      }
4069
      return tColDataPutValue(pColData, pData, nData);
×
4070
    }
4071
  } else {
4072
    return TSDB_CODE_INVALID_PARA;
×
4073
  }
4074
  return 0;
×
4075
}
4076
static int32_t tColDataUpdateValue72(SColData *pColData, uint8_t *pData, uint32_t nData, bool forward) {
3,595✔
4077
  uint8_t bv = GET_BIT2(pColData->pBitMap, pColData->nVal - 1);
3,595✔
4078
  if (bv == 0) {  // NONE ==> NULL
3,595!
4079
    pColData->numOfNone--;
3,595✔
4080
    pColData->nVal--;
3,595✔
4081
    if (!IS_VAR_DATA_TYPE(pColData->type)) {
3,595!
4082
      pColData->nData -= TYPE_BYTES[pColData->type];
2,543✔
4083
    }
4084
    if (pColData->numOfNone) {
3,595!
4085
      return tColDataAppendValue72(pColData, pData, nData);
×
4086
    } else {
4087
      for (int32_t iVal = 0; iVal < pColData->nVal; ++iVal) {
114,270✔
4088
        SET_BIT1(pColData->pBitMap, iVal, GET_BIT2(pColData->pBitMap, iVal) - 1);
110,675✔
4089
      }
4090
      pColData->flag = (HAS_VALUE | HAS_NULL);
3,595✔
4091
      return tColDataAppendValue62(pColData, pData, nData);
3,594✔
4092
    }
4093
  } else if (bv == 2 && forward) {  // VALUE ==> NULL
×
4094
    pColData->numOfValue--;
×
4095
    pColData->nVal--;
×
4096
    if (pColData->numOfValue) {
×
4097
      if (IS_VAR_DATA_TYPE(pColData->type)) {
×
4098
        pColData->nData = pColData->aOffset[pColData->nVal];
×
4099
      } else {
4100
        pColData->nData -= TYPE_BYTES[pColData->type];
×
4101
      }
4102
      return tColDataAppendValue72(pColData, pData, nData);
×
4103
    } else {
4104
      for (int32_t iVal = 0; iVal < pColData->nVal; ++iVal) {
×
4105
        SET_BIT1(pColData->pBitMap, iVal, GET_BIT2(pColData->pBitMap, iVal));
×
4106
      }
4107
      pColData->flag = (HAS_NULL | HAS_NONE);
×
4108
      pColData->nData = 0;
×
4109
      return tColDataAppendValue32(pColData, pData, nData);
×
4110
    }
4111
  }
4112
  return 0;
×
4113
}
4114
static FORCE_INLINE int32_t tColDataUpdateNothing(SColData *pColData, uint8_t *pData, uint32_t nData, bool forward) {
13,175✔
4115
  return 0;
13,175✔
4116
}
4117
static int32_t (*tColDataUpdateValueImpl[8][3])(SColData *pColData, uint8_t *pData, uint32_t nData, bool forward) = {
4118
    {NULL, NULL, NULL},                                                     // 0
4119
    {tColDataUpdateValue10, tColDataUpdateNothing, tColDataUpdateValue12},  // HAS_NONE
4120
    {tColDataUpdateValue20, tColDataUpdateNothing, tColDataUpdateNothing},  // HAS_NULL
4121
    {tColDataUpdateValue30, tColDataUpdateNothing, tColDataUpdateValue32},  // HAS_NULL|HAS_NONE
4122
    {tColDataUpdateValue40, tColDataUpdateNothing, tColDataUpdateValue42},  // HAS_VALUE
4123
    {tColDataUpdateValue50, tColDataUpdateNothing, tColDataUpdateValue52},  // HAS_VALUE|HAS_NONE
4124
    {tColDataUpdateValue60, tColDataUpdateNothing, tColDataUpdateValue62},  // HAS_VALUE|HAS_NULL
4125
    {tColDataUpdateValue70, tColDataUpdateNothing, tColDataUpdateValue72},  // HAS_VALUE|HAS_NULL|HAS_NONE
4126

4127
    //    VALUE             NONE        NULL
4128
};
4129
int32_t tColDataUpdateValue(SColData *pColData, SColVal *pColVal, bool forward) {
13,709,152✔
4130
  if (!(pColData->cid == pColVal->cid && pColData->type == pColVal->value.type)) return TSDB_CODE_INVALID_PARA;
13,709,152!
4131
  if (!(pColData->nVal > 0)) return TSDB_CODE_INVALID_PARA;
13,712,900!
4132

4133
  if (tColDataUpdateValueImpl[pColData->flag][pColVal->flag] == NULL) return 0;
13,712,900!
4134

4135
  return tColDataUpdateValueImpl[pColData->flag][pColVal->flag](
13,712,900✔
4136
      pColData, VALUE_GET_DATUM(&pColVal->value, pColData->type), pColVal->value.nData, forward);
13,712,900!
4137
}
4138

4139
static FORCE_INLINE void tColDataGetValue1(SColData *pColData, int32_t iVal, SColVal *pColVal) {  // HAS_NONE
125,366,866✔
4140
  *pColVal = COL_VAL_NONE(pColData->cid, pColData->type);
125,366,866✔
4141
}
125,366,866✔
4142
static FORCE_INLINE void tColDataGetValue2(SColData *pColData, int32_t iVal, SColVal *pColVal) {  // HAS_NULL
871,751✔
4143
  *pColVal = COL_VAL_NULL(pColData->cid, pColData->type);
871,751✔
4144
}
871,751✔
4145
static FORCE_INLINE void tColDataGetValue3(SColData *pColData, int32_t iVal,
11,104✔
4146
                                           SColVal *pColVal) {  // HAS_NULL|HAS_NONE
4147
  switch (GET_BIT1(pColData->pBitMap, iVal)) {
11,104!
4148
    case 0:
1,011✔
4149
      *pColVal = COL_VAL_NONE(pColData->cid, pColData->type);
1,011✔
4150
      break;
1,011✔
4151
    case 1:
10,095✔
4152
      *pColVal = COL_VAL_NULL(pColData->cid, pColData->type);
10,095✔
4153
      break;
10,095✔
4154
    default:
×
4155
      break;
×
4156
  }
4157
}
11,104✔
4158
static FORCE_INLINE void tColDataGetValue4(SColData *pColData, int32_t iVal, SColVal *pColVal) {  // HAS_VALUE
2,147,483,647✔
4159
  SValue value = {.type = pColData->type};
2,147,483,647✔
4160
  if (IS_VAR_DATA_TYPE(pColData->type)) {
2,147,483,647!
4161
    if (iVal + 1 < pColData->nVal) {
343,078,299!
4162
      value.nData = pColData->aOffset[iVal + 1] - pColData->aOffset[iVal];
366,490,419✔
4163
    } else {
4164
      value.nData = pColData->nData - pColData->aOffset[iVal];
×
4165
    }
4166
    value.pData = pColData->pData + pColData->aOffset[iVal];
343,078,299✔
4167
  } else {
4168
    valueSetDatum(&value, pColData->type, pColData->pData + tDataTypes[pColData->type].bytes * iVal,
2,147,483,647✔
4169
                  tDataTypes[pColData->type].bytes);
2,147,483,647✔
4170
  }
4171
  *pColVal = COL_VAL_VALUE(pColData->cid, value);
2,147,483,647✔
4172
}
285,905,428✔
4173
static FORCE_INLINE void tColDataGetValue5(SColData *pColData, int32_t iVal,
7,688,660✔
4174
                                           SColVal *pColVal) {  // HAS_VALUE|HAS_NONE
4175
  switch (GET_BIT1(pColData->pBitMap, iVal)) {
7,688,660!
4176
    case 0:
3,864,500✔
4177
      *pColVal = COL_VAL_NONE(pColData->cid, pColData->type);
3,864,500✔
4178
      break;
3,864,500✔
4179
    case 1:
3,951,196✔
4180
      tColDataGetValue4(pColData, iVal, pColVal);
4181
      break;
3,991,867✔
4182
    default:
×
4183
      break;
×
4184
  }
4185
}
7,729,331✔
4186
static FORCE_INLINE void tColDataGetValue6(SColData *pColData, int32_t iVal,
316,247,656✔
4187
                                           SColVal *pColVal) {  // HAS_VALUE|HAS_NULL
4188
  switch (GET_BIT1(pColData->pBitMap, iVal)) {
316,247,656!
4189
    case 0:
36,219,498✔
4190
      *pColVal = COL_VAL_NULL(pColData->cid, pColData->type);
36,219,498✔
4191
      break;
36,219,498✔
4192
    case 1:
280,682,500✔
4193
      tColDataGetValue4(pColData, iVal, pColVal);
4194
      break;
281,913,530✔
4195
    default:
×
4196
      break;
×
4197
  }
4198
}
317,478,686✔
4199
static FORCE_INLINE void tColDataGetValue7(SColData *pColData, int32_t iVal,
×
4200
                                           SColVal *pColVal) {  // HAS_VALUE|HAS_NULL|HAS_NONE
4201
  switch (GET_BIT2(pColData->pBitMap, iVal)) {
×
4202
    case 0:
×
4203
      *pColVal = COL_VAL_NONE(pColData->cid, pColData->type);
×
4204
      break;
×
4205
    case 1:
×
4206
      *pColVal = COL_VAL_NULL(pColData->cid, pColData->type);
×
4207
      break;
×
4208
    case 2:
×
4209
      tColDataGetValue4(pColData, iVal, pColVal);
4210
      break;
×
4211
    default:
×
4212
      break;
×
4213
  }
4214
}
×
4215
static void (*tColDataGetValueImpl[])(SColData *pColData, int32_t iVal, SColVal *pColVal) = {
4216
    NULL,               // 0
4217
    tColDataGetValue1,  // HAS_NONE
4218
    tColDataGetValue2,  // HAS_NULL
4219
    tColDataGetValue3,  // HAS_NULL | HAS_NONE
4220
    tColDataGetValue4,  // HAS_VALUE
4221
    tColDataGetValue5,  // HAS_VALUE | HAS_NONE
4222
    tColDataGetValue6,  // HAS_VALUE | HAS_NULL
4223
    tColDataGetValue7   // HAS_VALUE | HAS_NULL | HAS_NONE
4224
};
4225
int32_t tColDataGetValue(SColData *pColData, int32_t iVal, SColVal *pColVal) {
2,147,483,647✔
4226
  if (iVal < 0 || iVal >= pColData->nVal ||
2,147,483,647!
4227
      (pColData->flag <= 0 || pColData->flag >= sizeof(tColDataGetValueImpl) / POINTER_BYTES)) {
2,147,483,647!
4228
    return TSDB_CODE_INVALID_PARA;
×
4229
  }
4230
  tColDataGetValueImpl[pColData->flag](pColData, iVal, pColVal);
2,147,483,647✔
4231
  return TSDB_CODE_SUCCESS;
2,147,483,647✔
4232
}
4233

4234
uint8_t tColDataGetBitValue(const SColData *pColData, int32_t iVal) {
779,647,958✔
4235
  switch (pColData->flag) {
779,647,958!
4236
    case HAS_NONE:
×
4237
      return 0;
×
4238
    case HAS_NULL:
×
4239
      return 1;
×
4240
    case (HAS_NULL | HAS_NONE):
11,110✔
4241
      return GET_BIT1(pColData->pBitMap, iVal);
11,110✔
4242
    case HAS_VALUE:
×
4243
      return 2;
×
4244
    case (HAS_VALUE | HAS_NONE):
8,377,767✔
4245
      return (GET_BIT1(pColData->pBitMap, iVal)) ? 2 : 0;
8,377,767✔
4246
    case (HAS_VALUE | HAS_NULL):
772,144,046✔
4247
      return GET_BIT1(pColData->pBitMap, iVal) + 1;
772,144,046✔
4248
    case (HAS_VALUE | HAS_NULL | HAS_NONE):
×
4249
      return GET_BIT2(pColData->pBitMap, iVal);
×
4250
    default:
×
4251
      return 0;
×
4252
  }
4253
}
4254

4255
int32_t tColDataCopy(SColData *pColDataFrom, SColData *pColData, xMallocFn xMalloc, void *arg) {
3,882✔
4256
  int32_t code = 0;
3,882✔
4257

4258
  *pColData = *pColDataFrom;
3,882✔
4259

4260
  // bitmap
4261
  switch (pColData->flag) {
3,882!
4262
    case (HAS_NULL | HAS_NONE):
105✔
4263
    case (HAS_VALUE | HAS_NONE):
4264
    case (HAS_VALUE | HAS_NULL):
4265
      pColData->pBitMap = xMalloc(arg, BIT1_SIZE(pColData->nVal));
105✔
4266
      if (pColData->pBitMap == NULL) {
105!
4267
        code = TSDB_CODE_OUT_OF_MEMORY;
×
4268
        goto _exit;
×
4269
      }
4270
      (void)memcpy(pColData->pBitMap, pColDataFrom->pBitMap, BIT1_SIZE(pColData->nVal));
105✔
4271
      break;
105✔
4272
    case (HAS_VALUE | HAS_NULL | HAS_NONE):
×
4273
      pColData->pBitMap = xMalloc(arg, BIT2_SIZE(pColData->nVal));
×
4274
      if (pColData->pBitMap == NULL) {
×
4275
        code = TSDB_CODE_OUT_OF_MEMORY;
×
4276
        goto _exit;
×
4277
      }
4278
      (void)memcpy(pColData->pBitMap, pColDataFrom->pBitMap, BIT2_SIZE(pColData->nVal));
×
4279
      break;
×
4280
    default:
3,777✔
4281
      pColData->pBitMap = NULL;
3,777✔
4282
      break;
3,777✔
4283
  }
4284

4285
  // offset
4286
  if (IS_VAR_DATA_TYPE(pColData->type) && (pColData->flag & HAS_VALUE)) {
3,882!
4287
    pColData->aOffset = xMalloc(arg, pColData->nVal << 2);
728✔
4288
    if (pColData->aOffset == NULL) {
728!
4289
      code = TSDB_CODE_OUT_OF_MEMORY;
×
4290
      goto _exit;
×
4291
    }
4292
    (void)memcpy(pColData->aOffset, pColDataFrom->aOffset, pColData->nVal << 2);
728✔
4293
  } else {
4294
    pColData->aOffset = NULL;
3,154✔
4295
  }
4296

4297
  // value
4298
  if (pColData->nData) {
3,882✔
4299
    pColData->pData = xMalloc(arg, pColData->nData);
3,722✔
4300
    if (pColData->pData == NULL) {
3,722!
4301
      code = TSDB_CODE_OUT_OF_MEMORY;
×
4302
      goto _exit;
×
4303
    }
4304

4305
    (void)memcpy(pColData->pData, pColDataFrom->pData, pColData->nData);
3,722✔
4306
  } else {
4307
    pColData->pData = NULL;
160✔
4308
  }
4309

4310
_exit:
3,882✔
4311
  return code;
3,882✔
4312
}
4313

4314
int32_t tColDataCompress(SColData *colData, SColDataCompressInfo *info, SBuffer *output, SBuffer *assist) {
3,195,201✔
4315
  int32_t code;
4316
  SBuffer local;
4317

4318
  if (!(colData->nVal > 0)) {
3,195,201!
4319
    return TSDB_CODE_INVALID_PARA;
×
4320
  }
4321

4322
  (*info) = (SColDataCompressInfo){
3,195,201✔
4323
      .cmprAlg = info->cmprAlg,
3,195,201✔
4324
      .columnFlag = colData->cflag,
3,195,201✔
4325
      .flag = colData->flag,
3,195,201✔
4326
      .dataType = colData->type,
3,195,201✔
4327
      .columnId = colData->cid,
3,195,201✔
4328
      .numOfData = colData->nVal,
3,195,201✔
4329
  };
4330

4331
  if (colData->flag == HAS_NONE || colData->flag == HAS_NULL) {
3,195,201!
4332
    return 0;
14,382✔
4333
  }
4334

4335
  tBufferInit(&local);
4336
  if (assist == NULL) {
3,180,819!
4337
    assist = &local;
×
4338
  }
4339

4340
  // bitmap
4341
  if (colData->flag != HAS_VALUE) {
3,180,819✔
4342
    if (colData->flag == (HAS_NONE | HAS_NULL | HAS_VALUE)) {
323,931✔
4343
      info->bitmapOriginalSize = BIT2_SIZE(colData->nVal);
32,415✔
4344
    } else {
4345
      info->bitmapOriginalSize = BIT1_SIZE(colData->nVal);
291,516✔
4346
    }
4347

4348
    SCompressInfo cinfo = {
323,931✔
4349
        .dataType = TSDB_DATA_TYPE_TINYINT,
4350
        .cmprAlg = info->cmprAlg,
323,931✔
4351
        .originalSize = info->bitmapOriginalSize,
323,931✔
4352
    };
4353

4354
    code = tCompressDataToBuffer(colData->pBitMap, &cinfo, output, assist);
323,931✔
4355
    if (code) {
323,968!
4356
      tBufferDestroy(&local);
4357
      return code;
×
4358
    }
4359

4360
    info->bitmapCompressedSize = cinfo.compressedSize;
323,968✔
4361
  }
4362

4363
  if (colData->flag == (HAS_NONE | HAS_NULL)) {
3,180,856✔
4364
    tBufferDestroy(&local);
4365
    return 0;
10,597✔
4366
  }
4367

4368
  // offset
4369
  if (IS_VAR_DATA_TYPE(colData->type)) {
3,170,259✔
4370
    info->offsetOriginalSize = sizeof(int32_t) * info->numOfData;
133,116✔
4371

4372
    SCompressInfo cinfo = {
133,116✔
4373
        .dataType = TSDB_DATA_TYPE_INT,
4374
        .cmprAlg = info->cmprAlg,
133,116✔
4375
        .originalSize = info->offsetOriginalSize,
133,116✔
4376
    };
4377

4378
    code = tCompressDataToBuffer(colData->aOffset, &cinfo, output, assist);
133,116✔
4379
    if (code) {
133,038!
4380
      tBufferDestroy(&local);
4381
      return code;
×
4382
    }
4383

4384
    info->offsetCompressedSize = cinfo.compressedSize;
133,038✔
4385
  }
4386

4387
  // data
4388
  if (colData->nData > 0) {
3,170,181!
4389
    info->dataOriginalSize = colData->nData;
3,170,193✔
4390

4391
    SCompressInfo cinfo = {
3,170,193✔
4392
        .dataType = colData->type,
3,170,193✔
4393
        .cmprAlg = info->cmprAlg,
3,170,193✔
4394
        .originalSize = info->dataOriginalSize,
3,170,193✔
4395
    };
4396

4397
    code = tCompressDataToBuffer(colData->pData, &cinfo, output, assist);
3,170,193✔
4398
    if (code) {
3,170,380!
4399
      tBufferDestroy(&local);
4400
      return code;
×
4401
    }
4402

4403
    info->dataCompressedSize = cinfo.compressedSize;
3,170,380✔
4404
  }
4405

4406
  tBufferDestroy(&local);
4407
  return 0;
3,170,368✔
4408
}
4409

4410
int32_t tColDataDecompress(void *input, SColDataCompressInfo *info, SColData *colData, SBuffer *assist) {
33,740,351✔
4411
  int32_t  code;
4412
  SBuffer  local;
4413
  uint8_t *data = (uint8_t *)input;
33,740,351✔
4414

4415
  tBufferInit(&local);
4416
  if (assist == NULL) {
33,740,351!
4417
    assist = &local;
×
4418
  }
4419

4420
  tColDataClear(colData);
33,740,351✔
4421
  colData->cid = info->columnId;
33,738,601✔
4422
  colData->type = info->dataType;
33,738,601✔
4423
  colData->cflag = info->columnFlag;
33,738,601✔
4424
  colData->nVal = info->numOfData;
33,738,601✔
4425
  colData->flag = info->flag;
33,738,601✔
4426

4427
  if (info->flag == HAS_NONE || info->flag == HAS_NULL) {
33,738,601✔
4428
    goto _exit;
3,987,155✔
4429
  }
4430

4431
  // bitmap
4432
  if (info->bitmapOriginalSize > 0) {
29,751,446✔
4433
    SCompressInfo cinfo = {
1,161,156✔
4434
        .dataType = TSDB_DATA_TYPE_TINYINT,
4435
        .cmprAlg = info->cmprAlg,
1,161,156✔
4436
        .originalSize = info->bitmapOriginalSize,
1,161,156✔
4437
        .compressedSize = info->bitmapCompressedSize,
1,161,156✔
4438
    };
4439

4440
    code = tRealloc(&colData->pBitMap, cinfo.originalSize);
1,161,156!
4441
    if (code) {
1,160,616!
4442
      tBufferDestroy(&local);
4443
      return code;
×
4444
    }
4445

4446
    code = tDecompressData(data, &cinfo, colData->pBitMap, cinfo.originalSize, assist);
1,160,616✔
4447
    if (code) {
1,161,211!
4448
      tBufferDestroy(&local);
4449
      return code;
×
4450
    }
4451

4452
    data += cinfo.compressedSize;
1,161,211✔
4453
  }
4454

4455
  if (info->flag == (HAS_NONE | HAS_NULL)) {
29,751,501✔
4456
    goto _exit;
602✔
4457
  }
4458

4459
  // offset
4460
  if (info->offsetOriginalSize > 0) {
29,750,899✔
4461
    SCompressInfo cinfo = {
5,277,614✔
4462
        .cmprAlg = info->cmprAlg,
5,277,614✔
4463
        .dataType = TSDB_DATA_TYPE_INT,
4464
        .originalSize = info->offsetOriginalSize,
5,277,614✔
4465
        .compressedSize = info->offsetCompressedSize,
5,277,614✔
4466
    };
4467

4468
    code = tRealloc((uint8_t **)&colData->aOffset, cinfo.originalSize);
5,277,614!
4469
    if (code) {
5,277,623!
4470
      tBufferDestroy(&local);
4471
      return code;
×
4472
    }
4473

4474
    code = tDecompressData(data, &cinfo, colData->aOffset, cinfo.originalSize, assist);
5,277,623✔
4475
    if (code) {
5,277,786!
4476
      tBufferDestroy(&local);
4477
      return code;
×
4478
    }
4479

4480
    data += cinfo.compressedSize;
5,277,786✔
4481
  }
4482

4483
  // data
4484
  if (info->dataOriginalSize > 0) {
29,751,071✔
4485
    colData->nData = info->dataOriginalSize;
29,749,019✔
4486

4487
    SCompressInfo cinfo = {
29,749,019✔
4488
        .cmprAlg = info->cmprAlg,
29,749,019✔
4489
        .dataType = colData->type,
29,749,019✔
4490
        .originalSize = info->dataOriginalSize,
29,749,019✔
4491
        .compressedSize = info->dataCompressedSize,
29,749,019✔
4492
    };
4493

4494
    code = tRealloc((uint8_t **)&colData->pData, cinfo.originalSize);
29,749,019!
4495
    if (code) {
29,748,137!
4496
      tBufferDestroy(&local);
4497
      return code;
×
4498
    }
4499

4500
    code = tDecompressData(data, &cinfo, colData->pData, cinfo.originalSize, assist);
29,748,137✔
4501
    if (code) {
29,746,925!
4502
      tBufferDestroy(&local);
4503
      return code;
×
4504
    }
4505

4506
    data += cinfo.compressedSize;
29,746,925✔
4507
  }
4508

4509
_exit:
2,052✔
4510
  switch (colData->flag) {
33,736,734✔
4511
    case HAS_NONE:
3,662,841✔
4512
      colData->numOfNone = colData->nVal;
3,662,841✔
4513
      break;
3,662,841✔
4514
    case HAS_NULL:
329,289✔
4515
      colData->numOfNull = colData->nVal;
329,289✔
4516
      break;
329,289✔
4517
    case HAS_VALUE:
28,584,990✔
4518
      colData->numOfValue = colData->nVal;
28,584,990✔
4519
      break;
28,584,990✔
4520
    default:
1,159,614✔
4521
      for (int32_t i = 0; i < colData->nVal; i++) {
450,150,276✔
4522
        uint8_t bitValue = tColDataGetBitValue(colData, i);
449,002,190✔
4523
        if (bitValue == 0) {
448,990,662✔
4524
          colData->numOfNone++;
4,022,814✔
4525
        } else if (bitValue == 1) {
444,967,848✔
4526
          colData->numOfNull++;
80,295,860✔
4527
        } else {
4528
          colData->numOfValue++;
364,671,988✔
4529
        }
4530
      }
4531
  }
4532
  tBufferDestroy(&local);
4533
  return 0;
33,725,206✔
4534
}
4535

4536
int32_t tColDataAddValueByDataBlock(SColData *pColData, int8_t type, int32_t bytes, int32_t nRows, char *lengthOrbitmap,
542✔
4537
                                    char *data) {
4538
  int32_t code = 0;
542✔
4539
  if (data == NULL) {
542✔
4540
    if (pColData->cflag & COL_IS_KEY) {
13!
4541
      code = TSDB_CODE_PAR_PRIMARY_KEY_IS_NULL;
×
4542
    } else {
4543
      for (int32_t i = 0; i < nRows; ++i) {
28✔
4544
        code = tColDataAppendValueImpl[pColData->flag][CV_FLAG_NONE](pColData, NULL, 0);
15✔
4545
      }
4546
    }
4547
    goto _exit;
13✔
4548
  }
4549

4550
  if (IS_VAR_DATA_TYPE(type)) {  // var-length data type
529!
4551
    if (!IS_STR_DATA_BLOB(type)) {
92!
4552
      for (int32_t i = 0; i < nRows; ++i) {
265✔
4553
        int32_t offset = *((int32_t *)lengthOrbitmap + i);
173✔
4554
        if (offset == -1) {
173!
4555
          if (pColData->cflag & COL_IS_KEY) {
×
4556
            code = TSDB_CODE_PAR_PRIMARY_KEY_IS_NULL;
×
4557
            goto _exit;
×
4558
          }
4559
          if ((code = tColDataAppendValueImpl[pColData->flag][CV_FLAG_NULL](pColData, NULL, 0))) {
×
4560
            goto _exit;
×
4561
          }
4562
        } else {
4563
          if (varDataTLen(data + offset) > bytes) {
173!
4564
            uError("var data length invalid, varDataTLen(data + offset):%d > bytes:%d", (int)varDataTLen(data + offset),
×
4565
                   bytes);
4566
            code = TSDB_CODE_PAR_VALUE_TOO_LONG;
×
4567
            goto _exit;
×
4568
          }
4569
          code = tColDataAppendValueImpl[pColData->flag][CV_FLAG_VALUE](pColData, (uint8_t *)varDataVal(data + offset),
173✔
4570
                                                                        varDataLen(data + offset));
173✔
4571
        }
4572
      }
4573
    } else {
4574
      for (int32_t i = 0; i < nRows; ++i) {
×
4575
        int32_t offset = *((int32_t *)lengthOrbitmap + i);
×
4576
        if (offset == -1) {
×
4577
          if (pColData->cflag & COL_IS_KEY) {
×
4578
            code = TSDB_CODE_PAR_PRIMARY_KEY_IS_NULL;
×
4579
            goto _exit;
×
4580
          }
4581
          if ((code = tColDataAppendValueImpl[pColData->flag][CV_FLAG_NULL](pColData, NULL, 0))) {
×
4582
            goto _exit;
×
4583
          }
4584
        } else {
4585
          if (blobDataTLen(data + offset) > TSDB_MAX_BLOB_LEN) {
×
4586
            uError("var data length invalid, varDataTLen(data + offset):%d > bytes:%d",
×
4587
                   (int)blobDataTLen(data + offset), bytes);
4588
            code = TSDB_CODE_PAR_VALUE_TOO_LONG;
×
4589
            goto _exit;
×
4590
          }
4591
          code = tColDataAppendValueImpl[pColData->flag][CV_FLAG_VALUE](pColData, (uint8_t *)blobDataVal(data + offset),
×
4592
                                                                        blobDataLen(data + offset));
×
4593
        }
4594
      }
4595
    }
4596
  } else {  // fixed-length data type
4597
    bool allValue = true;
437✔
4598
    bool allNull = true;
437✔
4599
    for (int32_t i = 0; i < nRows; ++i) {
1,248✔
4600
      if (!BMIsNull(lengthOrbitmap, i)) {
811✔
4601
        allNull = false;
598✔
4602
      } else {
4603
        allValue = false;
213✔
4604
      }
4605
    }
4606
    if ((pColData->cflag & COL_IS_KEY) && !allValue) {
437!
4607
      code = TSDB_CODE_PAR_PRIMARY_KEY_IS_NULL;
×
4608
      goto _exit;
×
4609
    }
4610

4611
    if (allValue) {
437✔
4612
      // optimize (todo)
4613
      for (int32_t i = 0; i < nRows; ++i) {
927✔
4614
        code = tColDataAppendValueImpl[pColData->flag][CV_FLAG_VALUE](pColData, (uint8_t *)data + bytes * i, bytes);
584✔
4615
      }
4616
    } else if (allNull) {
94✔
4617
      // optimize (todo)
4618
      for (int32_t i = 0; i < nRows; ++i) {
249✔
4619
        code = tColDataAppendValueImpl[pColData->flag][CV_FLAG_NULL](pColData, NULL, 0);
165✔
4620
        if (code) goto _exit;
165!
4621
      }
4622
    } else {
4623
      for (int32_t i = 0; i < nRows; ++i) {
72✔
4624
        if (BMIsNull(lengthOrbitmap, i)) {
62✔
4625
          code = tColDataAppendValueImpl[pColData->flag][CV_FLAG_NULL](pColData, NULL, 0);
48✔
4626
          if (code) goto _exit;
48!
4627
        } else {
4628
          code = tColDataAppendValueImpl[pColData->flag][CV_FLAG_VALUE](pColData, (uint8_t *)data + bytes * i, bytes);
14✔
4629
        }
4630
      }
4631
    }
4632
  }
4633

4634
_exit:
10✔
4635
  return code;
542✔
4636
}
4637

4638
int32_t tColDataAddValueByBind(SColData *pColData, TAOS_MULTI_BIND *pBind, int32_t buffMaxLen, initGeosFn igeos,
1,776,572✔
4639
                               checkWKBGeometryFn cgeos) {
4640
  int32_t code = 0;
1,776,572✔
4641

4642
  if (!(pBind->num == 1 && pBind->is_null && *pBind->is_null)) {
1,776,572!
4643
    if (!(pColData->type == pBind->buffer_type)) {
1,775,737!
4644
      return TSDB_CODE_INVALID_PARA;
×
4645
    }
4646
  }
4647

4648
  if (IS_VAR_DATA_TYPE(pColData->type)) {  // var-length data type
1,776,572!
4649
    if (pColData->type == TSDB_DATA_TYPE_GEOMETRY) {
41,045✔
4650
      code = igeos();
3✔
4651
      if (code) {
3!
4652
        return code;
×
4653
      }
4654
    }
4655
    for (int32_t i = 0; i < pBind->num; ++i) {
99,589✔
4656
      if (pBind->is_null && pBind->is_null[i]) {
58,622✔
4657
        if (pColData->cflag & COL_IS_KEY) {
773!
4658
          code = TSDB_CODE_PAR_PRIMARY_KEY_IS_NULL;
×
4659
          goto _exit;
×
4660
        }
4661
        code = tColDataAppendValueImpl[pColData->flag][CV_FLAG_NULL](pColData, NULL, 0);
773✔
4662
        if (code) goto _exit;
773!
4663
      } else if (pBind->length[i] > buffMaxLen) {
57,849!
4664
        return TSDB_CODE_PAR_VALUE_TOO_LONG;
×
4665
      } else {
4666
        if (pColData->type == TSDB_DATA_TYPE_GEOMETRY) {
57,849✔
4667
          code = cgeos((char *)pBind->buffer + pBind->buffer_length * i, (size_t)pBind->length[i]);
5✔
4668
          if (code) {
5✔
4669
            uError("stmt col[%d] bind geometry wrong format", i);
1!
4670
            goto _exit;
1✔
4671
          }
4672
        }
4673
        code = tColDataAppendValueImpl[pColData->flag][CV_FLAG_VALUE](
57,848✔
4674
            pColData, (uint8_t *)pBind->buffer + pBind->buffer_length * i, pBind->length[i]);
57,848✔
4675
      }
4676
    }
4677
  } else {  // fixed-length data type
4678
    bool allValue;
4679
    bool allNull;
4680
    if (pBind->is_null) {
1,735,527✔
4681
      bool same = (memcmp(pBind->is_null, pBind->is_null + 1, pBind->num - 1) == 0);
20,400✔
4682
      allNull = (same && pBind->is_null[0] != 0);
20,400!
4683
      allValue = (same && pBind->is_null[0] == 0);
20,400!
4684
    } else {
4685
      allNull = false;
1,715,127✔
4686
      allValue = true;
1,715,127✔
4687
    }
4688

4689
    if ((pColData->cflag & COL_IS_KEY) && !allValue) {
1,735,527!
4690
      code = TSDB_CODE_PAR_PRIMARY_KEY_IS_NULL;
×
4691
      goto _exit;
×
4692
    }
4693

4694
    if (allValue) {
1,735,527✔
4695
      // optimize (todo)
4696
      for (int32_t i = 0; i < pBind->num; ++i) {
24,314,200✔
4697
        code = tColDataAppendValueImpl[pColData->flag][CV_FLAG_VALUE](
22,695,361✔
4698
            pColData, (uint8_t *)pBind->buffer + TYPE_BYTES[pColData->type] * i, pBind->buffer_length);
22,695,361✔
4699
      }
4700
    } else if (allNull) {
32,477✔
4701
      // optimize (todo)
4702
      for (int32_t i = 0; i < pBind->num; ++i) {
124✔
4703
        code = tColDataAppendValueImpl[pColData->flag][CV_FLAG_NULL](pColData, NULL, 0);
62✔
4704
        if (code) goto _exit;
62!
4705
      }
4706
    } else {
4707
      for (int32_t i = 0; i < pBind->num; ++i) {
32,415!
4708
        if (pBind->is_null[i]) {
×
4709
          code = tColDataAppendValueImpl[pColData->flag][CV_FLAG_NULL](pColData, NULL, 0);
×
4710
          if (code) goto _exit;
×
4711
        } else {
4712
          code = tColDataAppendValueImpl[pColData->flag][CV_FLAG_VALUE](
×
4713
              pColData, (uint8_t *)pBind->buffer + TYPE_BYTES[pColData->type] * i, pBind->buffer_length);
×
4714
        }
4715
      }
4716
    }
4717
  }
4718

4719
_exit:
32,415✔
4720
  return code;
1,692,284✔
4721
}
4722

4723
int32_t tColDataAddValueByBind2(SColData *pColData, TAOS_STMT2_BIND *pBind, int32_t buffMaxLen, initGeosFn igeos,
354✔
4724
                                checkWKBGeometryFn cgeos) {
4725
  int32_t code = 0;
354✔
4726

4727
  if (!(pBind->num == 1 && pBind->is_null && *pBind->is_null)) {
354!
4728
    if (!(pColData->type == pBind->buffer_type)) {
354!
4729
      return TSDB_CODE_INVALID_PARA;
×
4730
    }
4731
  }
4732

4733
  if (IS_VAR_DATA_TYPE(pColData->type)) {  // var-length data type
498!
4734
    if (pColData->type == TSDB_DATA_TYPE_GEOMETRY) {
144✔
4735
      code = igeos();
15✔
4736
      if (code) {
15!
4737
        return code;
×
4738
      }
4739
    }
4740

4741
    uint8_t *buf = pBind->buffer;
144✔
4742
    for (int32_t i = 0; i < pBind->num; ++i) {
537✔
4743
      if (pBind->is_null && pBind->is_null[i]) {
393✔
4744
        if (pColData->cflag & COL_IS_KEY) {
3!
4745
          code = TSDB_CODE_PAR_PRIMARY_KEY_IS_NULL;
×
4746
          goto _exit;
×
4747
        }
4748
        if (pBind->is_null[i] == 1) {
3!
4749
          code = tColDataAppendValueImpl[pColData->flag][CV_FLAG_NULL](pColData, NULL, 0);
3✔
4750
          if (code) goto _exit;
3!
4751
        } else {
4752
          code = tColDataAppendValueImpl[pColData->flag][CV_FLAG_NONE](pColData, NULL, 0);
×
4753
          if (code) goto _exit;
×
4754
        }
4755
      } else if (pBind->length[i] > buffMaxLen) {
390!
4756
        return TSDB_CODE_PAR_VALUE_TOO_LONG;
×
4757
      } else {
4758
        if (pColData->type == TSDB_DATA_TYPE_GEOMETRY) {
390✔
4759
          code = cgeos(buf, pBind->length[i]);
19✔
4760
          if (code) {
19!
4761
            uError("stmt2 col[%d] bind geometry wrong format", i);
×
4762
            goto _exit;
×
4763
          }
4764
        }
4765
        code = tColDataAppendValueImpl[pColData->flag][CV_FLAG_VALUE](pColData, buf, pBind->length[i]);
390✔
4766
        buf += pBind->length[i];
390✔
4767
      }
4768
    }
4769
  } else {  // fixed-length data type
4770
    bool allValue;
4771
    bool allNull;
4772
    bool allNone;
4773
    if (pBind->is_null) {
210✔
4774
      bool same = (memcmp(pBind->is_null, pBind->is_null + 1, pBind->num - 1) == 0);
1✔
4775
      allNull = (same && pBind->is_null[0] == 1);
1!
4776
      allNone = (same && pBind->is_null[0] > 1);
1!
4777
      allValue = (same && pBind->is_null[0] == 0);
1!
4778
    } else {
4779
      allNull = false;
209✔
4780
      allNone = false;
209✔
4781
      allValue = true;
209✔
4782
    }
4783

4784
    if ((pColData->cflag & COL_IS_KEY) && !allValue) {
210!
4785
      code = TSDB_CODE_PAR_PRIMARY_KEY_IS_NULL;
×
4786
      goto _exit;
×
4787
    }
4788

4789
    if (allValue) {
210!
4790
      // optimize (todo)
4791
      for (int32_t i = 0; i < pBind->num; ++i) {
654✔
4792
        uint8_t *val = (uint8_t *)pBind->buffer + TYPE_BYTES[pColData->type] * i;
444✔
4793
        if (TSDB_DATA_TYPE_BOOL == pColData->type && *val > 1) {
444!
4794
          *val = 1;
1✔
4795
        }
4796

4797
        code = tColDataAppendValueImpl[pColData->flag][CV_FLAG_VALUE](pColData, val, TYPE_BYTES[pColData->type]);
444✔
4798
      }
4799
    } else if (allNull) {
×
4800
      // optimize (todo)
4801
      for (int32_t i = 0; i < pBind->num; ++i) {
×
4802
        code = tColDataAppendValueImpl[pColData->flag][CV_FLAG_NULL](pColData, NULL, 0);
×
4803
        if (code) goto _exit;
×
4804
      }
4805
    } else if (allNone) {
×
4806
      // optimize (todo)
4807
      for (int32_t i = 0; i < pBind->num; ++i) {
×
4808
        code = tColDataAppendValueImpl[pColData->flag][CV_FLAG_NONE](pColData, NULL, 0);
×
4809
        if (code) goto _exit;
×
4810
      }
4811
    } else {
4812
      for (int32_t i = 0; i < pBind->num; ++i) {
×
4813
        if (pBind->is_null[i]) {
×
4814
          if (pBind->is_null[i] == 1) {
×
4815
            code = tColDataAppendValueImpl[pColData->flag][CV_FLAG_NULL](pColData, NULL, 0);
×
4816
            if (code) goto _exit;
×
4817
          } else {
4818
            code = tColDataAppendValueImpl[pColData->flag][CV_FLAG_NONE](pColData, NULL, 0);
×
4819
            if (code) goto _exit;
×
4820
          }
4821
        } else {
4822
          uint8_t *val = (uint8_t *)pBind->buffer + TYPE_BYTES[pColData->type] * i;
×
4823
          if (TSDB_DATA_TYPE_BOOL == pColData->type && *val > 1) {
×
4824
            *val = 1;
×
4825
          }
4826

4827
          code = tColDataAppendValueImpl[pColData->flag][CV_FLAG_VALUE](pColData, val, TYPE_BYTES[pColData->type]);
×
4828
        }
4829
      }
4830
    }
4831
  }
4832

4833
_exit:
×
4834
  return code;
354✔
4835
}
4836
int32_t tColDataAddValueByBind2WithBlob(SColData *pColData, TAOS_STMT2_BIND *pBind, int32_t buffMaxLen,
×
4837
                                        SBlobSet *pBlobSet) {
4838
  int32_t code = 0;
×
4839

4840
  if (!(pBind->num == 1 && pBind->is_null && *pBind->is_null)) {
×
4841
    if (!(pColData->type == pBind->buffer_type)) {
×
4842
      return TSDB_CODE_INVALID_PARA;
×
4843
    }
4844
  }
4845

4846
  if (IS_STR_DATA_BLOB(pColData->type)) {  // var-length data type
×
4847
    uint8_t *buf = pBind->buffer;
×
4848
    for (int32_t i = 0; i < pBind->num; ++i) {
×
4849
      if (pBind->is_null && pBind->is_null[i]) {
×
4850
        if (pColData->cflag & COL_IS_KEY) {
×
4851
          code = TSDB_CODE_PAR_PRIMARY_KEY_IS_NULL;
×
4852
          goto _exit;
×
4853
        }
4854
        if (pBind->is_null[i] == 1) {
×
4855
          code = tColDataAppendValueBlobImpl[pColData->flag][CV_FLAG_NULL](pBlobSet, pColData, NULL, 0);
×
4856
          if (code) goto _exit;
×
4857
        } else {
4858
          code = tColDataAppendValueBlobImpl[pColData->flag][CV_FLAG_NONE](pBlobSet, pColData, NULL, 0);
×
4859
          if (code) goto _exit;
×
4860
        }
4861
      } else if (pBind->length[i] > buffMaxLen) {
×
4862
        return TSDB_CODE_PAR_VALUE_TOO_LONG;
×
4863
      } else {
4864
        code = tColDataAppendValueBlobImpl[pColData->flag][CV_FLAG_VALUE](pBlobSet, pColData, buf, pBind->length[i]);
×
4865
        buf += pBind->length[i];
×
4866
      }
4867
    }
4868
  }
4869
_exit:
×
4870
  return code;
×
4871
}
4872

4873
/* build rows to `rowArray` from bind
4874
 * `infos` is the bind information array
4875
 * `numOfInfos` is the number of bind information
4876
 * `infoSorted` is whether the bind information is sorted by column id
4877
 * `pTSchema` is the schema of the table
4878
 * `rowArray` is the array to store the rows
4879
 * `pOrdered` is the pointer to store ordered
4880
 * `pDupTs` is the pointer to store duplicateTs
4881
 */
4882
int32_t tRowBuildFromBind2(SBindInfo2 *infos, int32_t numOfInfos, SSHashObj *parsedCols, bool infoSorted,
7,499✔
4883
                           const STSchema *pTSchema, SArray *rowArray, bool *pOrdered, bool *pDupTs) {
4884
  if (infos == NULL || numOfInfos <= 0 || numOfInfos > pTSchema->numOfCols || pTSchema == NULL || rowArray == NULL) {
7,499!
4885
    return TSDB_CODE_INVALID_PARA;
×
4886
  }
4887
  int8_t hasBlob = schemaHasBlob(pTSchema);
7,510✔
4888
  if (!infoSorted) {
7,513✔
4889
    taosqsort_r(infos, numOfInfos, sizeof(SBindInfo2), NULL, tBindInfoCompare);
7✔
4890
  }
4891

4892
  int32_t code = 0;
7,513✔
4893
  int32_t numOfRows = infos[0].bind->num;
7,513✔
4894
  SArray *colValArray, *bufArray;
4895
  SColVal colVal;
4896
  int32_t numOfFixedValue = 0;
7,513✔
4897

4898
  if ((colValArray = taosArrayInit(numOfInfos, sizeof(SColVal))) == NULL) {
7,513!
4899
    return terrno;
×
4900
  }
4901
  if ((bufArray = taosArrayInit(numOfInfos, sizeof(uint8_t *))) == NULL) {
7,503!
4902
    taosArrayDestroy(colValArray);
×
4903
    return terrno;
×
4904
  }
4905
  for (int i = 0; i < numOfInfos; ++i) {
118,128✔
4906
    if (parsedCols) {
110,636✔
4907
      SColVal *pParsedVal = tSimpleHashGet(parsedCols, &infos[i].columnId, sizeof(int16_t));
132✔
4908
      if (pParsedVal) {
132✔
4909
        continue;
24✔
4910
      }
4911
    }
4912
    if (!taosArrayPush(bufArray, &infos[i].bind->buffer)) {
221,207!
4913
      taosArrayDestroy(colValArray);
×
4914
      taosArrayDestroy(bufArray);
×
4915
      return terrno;
×
4916
    }
4917
  }
4918

4919
  SRowKey rowKey, lastRowKey;
4920
  for (int32_t iRow = 0; iRow < numOfRows; iRow++) {
255,072✔
4921
    taosArrayClear(colValArray);
250,851✔
4922

4923
    for (int32_t iInfo = 0; iInfo < numOfInfos; iInfo++) {
1,385,530✔
4924
      if (parsedCols) {
1,123,435✔
4925
        SColVal *pParsedVal = tSimpleHashGet(parsedCols, &infos[iInfo].columnId, sizeof(int16_t));
132✔
4926
        if (pParsedVal) {
132!
4927
          numOfFixedValue++;
×
4928
          colVal = *pParsedVal;
×
4929

4930
          if (taosArrayPush(colValArray, &colVal) == NULL) {
24!
4931
            if (IS_VAR_DATA_TYPE(pParsedVal->value.type)) {
×
4932
              taosMemoryFree(colVal.value.pData);
×
4933
            }
4934
            code = terrno;
×
4935
            goto _exit;
1✔
4936
          }
4937
          continue;
24✔
4938
        }
4939
      }
4940

4941
      if (infos[iInfo].bind->is_null && infos[iInfo].bind->is_null[iRow]) {
1,127,244✔
4942
        if (infos[iInfo].bind->is_null[iRow] == 1) {
3!
4943
          if (iInfo == 0) {
3✔
4944
            code = TSDB_CODE_PAR_PRIMARY_KEY_IS_NULL;
1✔
4945
            goto _exit;
1✔
4946
          }
4947
          colVal = COL_VAL_NULL(infos[iInfo].columnId, infos[iInfo].type);
2✔
4948
        } else {
4949
          colVal = COL_VAL_NONE(infos[iInfo].columnId, infos[iInfo].type);
×
4950
        }
4951
      } else {
4952
        SValue value = {
1,127,241✔
4953
            .type = infos[iInfo].type,
1,127,241✔
4954
        };
4955
        if (IS_VAR_DATA_TYPE(infos[iInfo].type)) {
1,127,241!
4956
          if (IS_STR_DATA_BLOB(infos[iInfo].type)) {
85,623!
4957
            int32_t   length = infos[iInfo].bind->length[iRow];
×
4958
            uint8_t **data = &((uint8_t **)TARRAY_DATA(bufArray))[iInfo - numOfFixedValue];
×
4959
            value.nData = length;
×
4960
            if (value.nData > (TSDB_MAX_BLOB_LEN - BLOBSTR_HEADER_SIZE)) {
×
4961
              code = TSDB_CODE_PAR_VALUE_TOO_LONG;
×
4962
              uError("stmt bind param[%d] length:%d  greater than type maximum lenght: %d", iInfo,
×
4963
                     value.nData + (uint32_t)(BLOBSTR_HEADER_SIZE), infos[iInfo].bytes);
4964
              goto _exit;
×
4965
            }
4966
            value.pData = *data;
×
4967
            *data += length;
×
4968
          } else {
4969
            int32_t   length = infos[iInfo].bind->length[iRow];
85,623✔
4970
            uint8_t **data = &((uint8_t **)TARRAY_DATA(bufArray))[iInfo - numOfFixedValue];
85,623✔
4971
            value.nData = length;
85,623✔
4972
            if (value.nData > infos[iInfo].bytes - VARSTR_HEADER_SIZE) {
85,623!
4973
              code = TSDB_CODE_PAR_VALUE_TOO_LONG;
×
4974
              uError("stmt bind param[%d] length:%d  greater than type maximum lenght: %d", iInfo,
×
4975
                     value.nData + (uint32_t)(VARSTR_HEADER_SIZE), infos[iInfo].bytes);
4976
              goto _exit;
×
4977
            }
4978
            value.pData = *data;
85,623✔
4979
            *data += length;
85,623✔
4980
          }
4981
          // value.pData = (uint8_t *)infos[iInfo].bind->buffer + infos[iInfo].bind->buffer_length * iRow;
4982
        } else {
4983
          uint8_t *val = (uint8_t *)infos[iInfo].bind->buffer + infos[iInfo].bytes * iRow;
1,041,618✔
4984
          if (TSDB_DATA_TYPE_BOOL == value.type && *val > 1) {
1,041,618✔
4985
            *val = 1;
1✔
4986
          }
4987
          valueSetDatum(&value, infos[iInfo].type, val, infos[iInfo].bytes);
1,041,618✔
4988
        }
4989
        colVal = COL_VAL_VALUE(infos[iInfo].columnId, value);
1,138,469✔
4990
      }
4991
      if (taosArrayPush(colValArray, &colVal) == NULL) {
1,134,802!
4992
        code = terrno;
×
4993
        goto _exit;
×
4994
      }
4995
    }
4996

4997
    SRow *row;
4998

4999
    if (hasBlob == 0) {
262,095✔
5000
      SRowBuildScanInfo sinfo = {0};
247,668✔
5001
      if ((code = tRowBuild(colValArray, pTSchema, &row, &sinfo))) {
247,668!
5002
        goto _exit;
×
5003
      }
5004
    } else {
5005
      SRowBuildScanInfo sinfo = {.hasBlob = 1, .scanType = ROW_BUILD_UPDATE};
14,427✔
5006
      if ((code = tRowBuildWithBlob(colValArray, pTSchema, &row, NULL, &sinfo))) {
14,427!
5007
        goto _exit;
×
5008
      }
5009
    }
5010

5011
    if ((taosArrayPush(rowArray, &row)) == NULL) {
247,594!
5012
      code = terrno;
×
5013
      goto _exit;
×
5014
    }
5015

5016
    if (pOrdered && pDupTs) {
247,594!
5017
      tRowGetKey(row, &rowKey);
495,290✔
5018
      if (iRow == 0) {
247,631✔
5019
        *pOrdered = true;
7,509✔
5020
        *pDupTs = false;
7,509✔
5021
      } else {
5022
        if (*pOrdered) {
240,122!
5023
          int32_t res = tRowKeyCompare(&rowKey, &lastRowKey);
240,168✔
5024
          *pOrdered = (res >= 0);
240,168✔
5025
          if (!*pDupTs) {
240,168!
5026
            *pDupTs = (res == 0);
240,233✔
5027
          }
5028
        }
5029
      }
5030
      lastRowKey = rowKey;
247,631✔
5031
    }
5032
  }
5033
_exit:
4,221✔
5034
  if (code != 0) {
4,222✔
5035
    uError("tRowBuildFromBind2 failed, code=%d", code);
1!
5036
  }
5037
  taosArrayDestroy(colValArray);
4,222✔
5038
  taosArrayDestroy(bufArray);
7,508✔
5039
  return code;
7,523✔
5040
}
5041
/* build rows to `rowArray` from bind
5042
 * `infos` is the bind information array
5043
 * `numOfInfos` is the number of bind information
5044
 * `infoSorted` is whether the bind information is sorted by column id
5045
 * `pTSchema` is the schema of the table
5046
 * `rowArray` is the array to store the rows
5047
 * `pOrdered` is the pointer to store ordered
5048
 * `pDupTs` is the pointer to store duplicateTs
5049
 */
5050
int32_t tRowBuildFromBind2WithBlob(SBindInfo2 *infos, int32_t numOfInfos, bool infoSorted, const STSchema *pTSchema,
×
5051
                                   SArray *rowArray, bool *pOrdered, bool *pDupTs, SBlobSet *pBlobSet) {
5052
  if (infos == NULL || numOfInfos <= 0 || numOfInfos > pTSchema->numOfCols || pTSchema == NULL || rowArray == NULL) {
×
5053
    return TSDB_CODE_INVALID_PARA;
×
5054
  }
5055
  int8_t hasBlob = schemaHasBlob(pTSchema);
×
5056
  if (!infoSorted) {
×
5057
    taosqsort_r(infos, numOfInfos, sizeof(SBindInfo2), NULL, tBindInfoCompare);
×
5058
  }
5059

5060
  int32_t code = 0;
×
5061
  int32_t numOfRows = infos[0].bind->num;
×
5062
  SArray *colValArray, *bufArray;
5063
  SColVal colVal;
5064

5065
  if ((colValArray = taosArrayInit(numOfInfos, sizeof(SColVal))) == NULL) {
×
5066
    return terrno;
×
5067
  }
5068
  if ((bufArray = taosArrayInit(numOfInfos, sizeof(uint8_t *))) == NULL) {
×
5069
    taosArrayDestroy(colValArray);
×
5070
    return terrno;
×
5071
  }
5072
  for (int i = 0; i < numOfInfos; ++i) {
×
5073
    if (!taosArrayPush(bufArray, &infos[i].bind->buffer)) {
×
5074
      taosArrayDestroy(colValArray);
×
5075
      taosArrayDestroy(bufArray);
×
5076
      return terrno;
×
5077
    }
5078
  }
5079

5080
  SRowKey rowKey, lastRowKey;
5081
  for (int32_t iRow = 0; iRow < numOfRows; iRow++) {
×
5082
    taosArrayClear(colValArray);
×
5083

5084
    for (int32_t iInfo = 0; iInfo < numOfInfos; iInfo++) {
×
5085
      if (infos[iInfo].bind->is_null && infos[iInfo].bind->is_null[iRow]) {
×
5086
        if (infos[iInfo].bind->is_null[iRow] == 1) {
×
5087
          if (iInfo == 0) {
×
5088
            code = TSDB_CODE_PAR_PRIMARY_KEY_IS_NULL;
×
5089
            goto _exit;
×
5090
          }
5091
          colVal = COL_VAL_NULL(infos[iInfo].columnId, infos[iInfo].type);
×
5092
        } else {
5093
          colVal = COL_VAL_NONE(infos[iInfo].columnId, infos[iInfo].type);
×
5094
        }
5095
      } else {
5096
        SValue value = {
×
5097
            .type = infos[iInfo].type,
×
5098
        };
5099
        if (IS_VAR_DATA_TYPE(infos[iInfo].type)) {
×
5100
          if (IS_STR_DATA_BLOB(infos[iInfo].type)) {
×
5101
            int32_t   length = infos[iInfo].bind->length[iRow];
×
5102
            uint8_t **data = &((uint8_t **)TARRAY_DATA(bufArray))[iInfo];
×
5103
            value.nData = length;
×
5104
            if (value.nData > (TSDB_MAX_BLOB_LEN - BLOBSTR_HEADER_SIZE)) {
×
5105
              code = TSDB_CODE_PAR_VALUE_TOO_LONG;
×
5106
              uError("stmt bind param[%d] length:%d  greater than type maximum lenght: %d", iInfo, value.nData,
×
5107
                     pTSchema->columns[infos[iInfo].columnId - 1].bytes);
5108
              goto _exit;
×
5109
            }
5110
            value.pData = *data;
×
5111
            *data += length;
×
5112
          } else {
5113
            int32_t   length = infos[iInfo].bind->length[iRow];
×
5114
            uint8_t **data = &((uint8_t **)TARRAY_DATA(bufArray))[iInfo];
×
5115
            value.nData = length;
×
5116
            if (value.nData > pTSchema->columns[infos[iInfo].columnId - 1].bytes - VARSTR_HEADER_SIZE) {
×
5117
              code = TSDB_CODE_PAR_VALUE_TOO_LONG;
×
5118
              uError("stmt bind param[%d] length:%d  greater than type maximum lenght: %d", iInfo, value.nData,
×
5119
                     pTSchema->columns[infos[iInfo].columnId - 1].bytes);
5120
              goto _exit;
×
5121
            }
5122
            value.pData = *data;
×
5123
            *data += length;
×
5124
          }
5125

5126
          // value.pData = (uint8_t *)infos[iInfo].bind->buffer + infos[iInfo].bind->buffer_length * iRow;
5127
        } else {
5128
          uint8_t *val = (uint8_t *)infos[iInfo].bind->buffer + infos[iInfo].bytes * iRow;
×
5129
          if (TSDB_DATA_TYPE_BOOL == value.type && *val > 1) {
×
5130
            *val = 1;
×
5131
          }
5132
          valueSetDatum(&value, infos[iInfo].type, val, infos[iInfo].bytes);
×
5133
        }
5134
        colVal = COL_VAL_VALUE(infos[iInfo].columnId, value);
×
5135
      }
5136
      if (taosArrayPush(colValArray, &colVal) == NULL) {
×
5137
        code = terrno;
×
5138
        goto _exit;
×
5139
      }
5140
    }
5141

5142
    SRow *row;
5143

5144
    if (hasBlob == 0) {
×
5145
      SRowBuildScanInfo sinfo = {0};
×
5146
      if ((code = tRowBuild(colValArray, pTSchema, &row, &sinfo))) {
×
5147
        goto _exit;
×
5148
      }
5149
    } else {
5150
      SRowBuildScanInfo sinfo = {.hasBlob = 1, .scanType = ROW_BUILD_UPDATE};
×
5151
      if ((code = tRowBuildWithBlob(colValArray, pTSchema, &row, pBlobSet, &sinfo))) {
×
5152
        goto _exit;
×
5153
      }
5154
    }
5155

5156
    if ((taosArrayPush(rowArray, &row)) == NULL) {
×
5157
      code = terrno;
×
5158
      goto _exit;
×
5159
    }
5160

5161
    if (pOrdered && pDupTs) {
×
5162
      tRowGetKey(row, &rowKey);
×
5163
      if (iRow == 0) {
×
5164
        *pOrdered = true;
×
5165
        *pDupTs = false;
×
5166
      } else {
5167
        if (*pOrdered) {
×
5168
          int32_t res = tRowKeyCompare(&rowKey, &lastRowKey);
×
5169
          *pOrdered = (res >= 0);
×
5170
          if (!*pDupTs) {
×
5171
            *pDupTs = (res == 0);
×
5172
          }
5173
        }
5174
      }
5175
      lastRowKey = rowKey;
×
5176
    }
5177
  }
5178
_exit:
×
5179
  taosArrayDestroy(colValArray);
×
5180
  taosArrayDestroy(bufArray);
×
5181
  return code;
×
5182
}
5183

5184
static int32_t tColDataCopyRowCell(SColData *pFromColData, int32_t iFromRow, SColData *pToColData, int32_t iToRow) {
374✔
5185
  int32_t code = TSDB_CODE_SUCCESS;
374✔
5186

5187
  if (IS_VAR_DATA_TYPE(pToColData->type)) {
374!
5188
    int32_t nData = (iFromRow < pFromColData->nVal - 1)
146✔
5189
                        ? pFromColData->aOffset[iFromRow + 1] - pFromColData->aOffset[iFromRow]
51✔
5190
                        : pFromColData->nData - pFromColData->aOffset[iFromRow];
73✔
5191
    if (iToRow == 0) {
73✔
5192
      pToColData->aOffset[iToRow] = 0;
14✔
5193
    }
5194

5195
    if (iToRow < pToColData->nVal - 1) {
73✔
5196
      pToColData->aOffset[iToRow + 1] = pToColData->aOffset[iToRow] + nData;
64✔
5197
    }
5198

5199
    (void)memcpy(pToColData->pData + pToColData->aOffset[iToRow], pFromColData->pData + pFromColData->aOffset[iFromRow],
73✔
5200
                 nData);
5201
  } else {
5202
    (void)memcpy(&pToColData->pData[TYPE_BYTES[pToColData->type] * iToRow],
301✔
5203
                 &pFromColData->pData[TYPE_BYTES[pToColData->type] * iFromRow], TYPE_BYTES[pToColData->type]);
301✔
5204
  }
5205
  return code;
374✔
5206
}
5207

5208
static int32_t tColDataCopyRowSingleCol(SColData *pFromColData, int32_t iFromRow, SColData *pToColData,
382✔
5209
                                        int32_t iToRow) {
5210
  int32_t code = TSDB_CODE_SUCCESS;
382✔
5211
  int     bit_val = 0;
382✔
5212

5213
  switch (pFromColData->flag) {
382!
5214
    case HAS_NONE: {
×
5215
      ROW_SET_BITMAP(pToColData->pBitMap, pToColData->flag, iToRow, BIT_FLG_NONE);
×
5216
    } break;
×
5217
    case HAS_NULL: {
8✔
5218
      ROW_SET_BITMAP(pToColData->pBitMap, pToColData->flag, iToRow, BIT_FLG_NULL);
8!
5219
    } break;
8✔
5220
    case (HAS_NULL | HAS_NONE): {
×
5221
      bit_val = GET_BIT1(pFromColData->pBitMap, iFromRow);
×
5222
      if (0 == bit_val)
×
5223
        ROW_SET_BITMAP(pToColData->pBitMap, pToColData->flag, iToRow, BIT_FLG_NONE);
×
5224
      else
5225
        ROW_SET_BITMAP(pToColData->pBitMap, pToColData->flag, iToRow, BIT_FLG_NULL);
×
5226
    } break;
×
5227
    case HAS_VALUE: {
328✔
5228
      ROW_SET_BITMAP(pToColData->pBitMap, pToColData->flag, iToRow, BIT_FLG_VALUE);
328!
5229
      TAOS_CHECK_RETURN(tColDataCopyRowCell(pFromColData, iFromRow, pToColData, iToRow));
328!
5230
    } break;
328✔
5231
    case (HAS_VALUE | HAS_NONE): {
×
5232
      bit_val = GET_BIT1(pFromColData->pBitMap, iFromRow);
×
5233
      if (0 == bit_val)
×
5234
        ROW_SET_BITMAP(pToColData->pBitMap, pToColData->flag, iToRow, BIT_FLG_NONE);
×
5235
      else
5236
        ROW_SET_BITMAP(pToColData->pBitMap, pToColData->flag, iToRow, BIT_FLG_VALUE);
×
5237
      TAOS_CHECK_RETURN(tColDataCopyRowCell(pFromColData, iFromRow, pToColData, iToRow));
×
5238
    } break;
×
5239
    case (HAS_VALUE | HAS_NULL): {
46✔
5240
      bit_val = GET_BIT1(pFromColData->pBitMap, iFromRow);
46✔
5241
      if (0 == bit_val)
46✔
5242
        ROW_SET_BITMAP(pToColData->pBitMap, pToColData->flag, iToRow, BIT_FLG_NULL);
28!
5243
      else
5244
        ROW_SET_BITMAP(pToColData->pBitMap, pToColData->flag, iToRow, BIT_FLG_VALUE);
18!
5245
      TAOS_CHECK_RETURN(tColDataCopyRowCell(pFromColData, iFromRow, pToColData, iToRow));
46!
5246
    } break;
46✔
5247
    case (HAS_VALUE | HAS_NULL | HAS_NONE): {
×
5248
      SET_BIT2(pToColData->pBitMap, iToRow, GET_BIT2(pFromColData->pBitMap, iFromRow));
×
5249
      TAOS_CHECK_RETURN(tColDataCopyRowCell(pFromColData, iFromRow, pToColData, iToRow));
×
5250
    } break;
×
5251
    default:
×
5252
      return -1;
×
5253
  }
5254

5255
  return code;
382✔
5256
}
5257

5258
static int32_t tColDataCopyRow(SColData *aFromColData, int32_t iFromRow, SColData *aToColData, int32_t iToRow,
89✔
5259
                               int32_t nColData) {
5260
  int32_t code = TSDB_CODE_SUCCESS;
89✔
5261

5262
  for (int32_t i = 0; i < nColData; i++) {
471✔
5263
    code = tColDataCopyRowSingleCol(&aFromColData[i], iFromRow, &aToColData[i], iToRow);
382✔
5264
    if (code != TSDB_CODE_SUCCESS) {
382!
5265
      return code;
×
5266
    }
5267
  }
5268

5269
  return code;
89✔
5270
}
5271

5272
static int32_t tColDataCopyRowAppend(SColData *aFromColData, int32_t iFromRow, SColData *aToColData, int32_t nColData) {
89✔
5273
  int32_t code = TSDB_CODE_SUCCESS;
89✔
5274

5275
  for (int32_t i = 0; i < nColData; i++) {
471✔
5276
    SColVal cv = {0};
382✔
5277
    code = tColDataGetValue(&aFromColData[i], iFromRow, &cv);
382✔
5278
    if (code != TSDB_CODE_SUCCESS) {
382!
5279
      return code;
×
5280
    }
5281
    code = tColDataAppendValue(&aToColData[i], &cv);
382✔
5282
    if (code != TSDB_CODE_SUCCESS) {
382!
5283
      return code;
×
5284
    }
5285
  }
5286

5287
  return code;
89✔
5288
}
5289

5290
void tColDataArrGetRowKey(SColData *aColData, int32_t nColData, int32_t iRow, SRowKey *key) {
10,167,735✔
5291
  SColVal cv;
5292

5293
  key->ts = ((TSKEY *)aColData[0].pData)[iRow];
10,167,735✔
5294
  key->numOfPKs = 0;
10,167,735✔
5295

5296
  for (int i = 1; i < nColData; i++) {
10,167,766!
5297
    if (aColData[i].cflag & COL_IS_KEY) {
10,179,519✔
5298
      tColDataGetValue4(&aColData[i], iRow, &cv);
31✔
5299
      key->pks[key->numOfPKs++] = cv.value;
31✔
5300
    } else {
5301
      break;
10,179,488✔
5302
    }
5303
  }
5304
}
10,167,735✔
5305

5306
static int32_t tColDataMergeSortMerge(SColData *aColData, int32_t start, int32_t mid, int32_t end, int32_t nColData) {
27✔
5307
  SColData *aDstColData = NULL;
27✔
5308
  int32_t   i = start, j = mid + 1, k = 0;
27✔
5309
  SRowKey   keyi, keyj;
5310

5311
  if (end > start) {
27!
5312
    aDstColData = taosMemoryCalloc(1, sizeof(SColData) * nColData);
27!
5313
    if (aDstColData == NULL) {
27!
5314
      return terrno;
×
5315
    }
5316
    for (int c = 0; c < nColData; ++c) {
141✔
5317
      tColDataInit(&aDstColData[c], aColData[c].cid, aColData[c].type, aColData[c].cflag);
114✔
5318
    }
5319
  }
5320

5321
  tColDataArrGetRowKey(aColData, nColData, i, &keyi);
27✔
5322
  tColDataArrGetRowKey(aColData, nColData, j, &keyj);
27✔
5323
  while (i <= mid && j <= end) {
76✔
5324
    if (tRowKeyCompare(&keyi, &keyj) <= 0) {
49✔
5325
      TAOS_CHECK_RETURN(tColDataCopyRowAppend(aColData, i++, aDstColData, nColData));
37!
5326
      tColDataArrGetRowKey(aColData, nColData, i, &keyi);
37✔
5327
    } else {
5328
      TAOS_CHECK_RETURN(tColDataCopyRowAppend(aColData, j++, aDstColData, nColData));
12!
5329
      if (j <= end) tColDataArrGetRowKey(aColData, nColData, j, &keyj);
12✔
5330
    }
5331
  }
5332

5333
  while (i <= mid) {
40✔
5334
    TAOS_CHECK_RETURN(tColDataCopyRowAppend(aColData, i++, aDstColData, nColData));
13!
5335
  }
5336

5337
  while (j <= end) {
54✔
5338
    TAOS_CHECK_RETURN(tColDataCopyRowAppend(aColData, j++, aDstColData, nColData));
27!
5339
  }
5340

5341
  for (i = start, k = 0; i <= end; ++i, ++k) {
116✔
5342
    TAOS_CHECK_RETURN(tColDataCopyRow(aDstColData, k, aColData, i, nColData));
89!
5343
  }
5344

5345
  if (aDstColData) {
27!
5346
    for (int32_t i = 0; i < nColData; i++) {
141✔
5347
      tColDataDestroy(&aDstColData[i]);
114✔
5348
    }
5349
    taosMemoryFree(aDstColData);
27!
5350
  }
5351

5352
  return TSDB_CODE_SUCCESS;
27✔
5353
}
5354

5355
static int32_t tColDataMergeSort(SColData *aColData, int32_t start, int32_t end, int32_t nColData) {
60✔
5356
  int32_t ret = TSDB_CODE_SUCCESS;
60✔
5357
  int32_t mid;
5358

5359
  if (start >= end) {
60✔
5360
    return TSDB_CODE_SUCCESS;
33✔
5361
  }
5362

5363
  mid = (start + end) / 2;
27✔
5364

5365
  ret = tColDataMergeSort(aColData, start, mid, nColData);
27✔
5366
  if (ret != TSDB_CODE_SUCCESS) {
27!
5367
    return ret;
×
5368
  }
5369

5370
  ret = tColDataMergeSort(aColData, mid + 1, end, nColData);
27✔
5371
  if (ret != TSDB_CODE_SUCCESS) {
27!
5372
    return ret;
×
5373
  }
5374

5375
  return tColDataMergeSortMerge(aColData, start, mid, end, nColData);
27✔
5376
}
5377

5378
static int32_t tColDataSort(SColData *aColData, int32_t nColData) {
6✔
5379
  int32_t nVal = aColData[0].nVal;
6✔
5380

5381
  if (nVal < 2) return TSDB_CODE_SUCCESS;
6!
5382

5383
  return tColDataMergeSort(aColData, 0, nVal - 1, nColData);
6✔
5384
}
5385

5386
static int32_t tColDataMerge(SArray **colArr) {
4✔
5387
  int32_t code = 0;
4✔
5388
  SArray *src = *colArr;
4✔
5389
  SArray *dst = NULL;
4✔
5390

5391
  dst = taosArrayInit(taosArrayGetSize(src), sizeof(SColData));
4✔
5392
  if (dst == NULL) {
4!
5393
    return terrno;
×
5394
  }
5395

5396
  for (int32_t i = 0; i < taosArrayGetSize(src); i++) {
15✔
5397
    SColData *srcCol = taosArrayGet(src, i);
11✔
5398

5399
    SColData *dstCol = taosArrayReserve(dst, 1);
11✔
5400
    if (dstCol == NULL) {
11!
5401
      code = terrno;
×
5402
      goto _exit;
×
5403
    }
5404
    tColDataInit(dstCol, srcCol->cid, srcCol->type, srcCol->cflag);
11✔
5405
  }
5406

5407
  int32_t numRows = ((SColData *)TARRAY_DATA(src))->nVal;
4✔
5408
  SRowKey lastKey;
5409
  for (int32_t i = 0; i < numRows; i++) {
16✔
5410
    SRowKey key;
5411
    tColDataArrGetRowKey((SColData *)TARRAY_DATA(src), taosArrayGetSize(src), i, &key);
12✔
5412

5413
    if (i == 0 || tRowKeyCompare(&key, &lastKey) != 0) {  // append new row
20✔
5414
      for (int32_t j = 0; j < taosArrayGetSize(src); j++) {
18✔
5415
        SColData *srcCol = taosArrayGet(src, j);
13✔
5416
        SColData *dstCol = taosArrayGet(dst, j);
13✔
5417

5418
        SColVal cv;
5419
        code = tColDataGetValue(srcCol, i, &cv);
13✔
5420
        if (code != TSDB_CODE_SUCCESS) {
13!
5421
          goto _exit;
×
5422
        }
5423
        code = tColDataAppendValue(dstCol, &cv);
13✔
5424
        if (code) {
13!
5425
          goto _exit;
×
5426
        }
5427
      }
5428
      lastKey = key;
5✔
5429
    } else {  // update existing row
5430
      for (int32_t j = 0; j < taosArrayGetSize(src); j++) {
24✔
5431
        SColData *srcCol = taosArrayGet(src, j);
17✔
5432
        SColData *dstCol = taosArrayGet(dst, j);
17✔
5433

5434
        SColVal cv;
5435
        code = tColDataGetValue(srcCol, i, &cv);
17✔
5436
        if (code != TSDB_CODE_SUCCESS) {
17!
5437
          goto _exit;
×
5438
        }
5439
        code = tColDataUpdateValue(dstCol, &cv, true);
17✔
5440
        if (code) {
17!
5441
          goto _exit;
×
5442
        }
5443
      }
5444
    }
5445
  }
5446

5447
_exit:
4✔
5448
  if (code) {
4!
5449
    taosArrayDestroyEx(dst, tColDataDestroy);
×
5450
  } else {
5451
    taosArrayDestroyEx(src, tColDataDestroy);
4✔
5452
    *colArr = dst;
4✔
5453
  }
5454
  return code;
4✔
5455
}
5456

5457
int32_t tColDataSortMerge(SArray **arr) {
12,191✔
5458
  SArray   *colDataArr = *arr;
12,191✔
5459
  int32_t   nColData = TARRAY_SIZE(colDataArr);
12,191✔
5460
  SColData *aColData = (SColData *)TARRAY_DATA(colDataArr);
12,191✔
5461

5462
  if (!(aColData[0].type == TSDB_DATA_TYPE_TIMESTAMP)) {
12,191!
5463
    return TSDB_CODE_PAR_INVALID_FIRST_COLUMN;
×
5464
  }
5465
  if (!(aColData[0].cid == PRIMARYKEY_TIMESTAMP_COL_ID)) {
12,191!
5466
    return TSDB_CODE_PAR_INVALID_FIRST_COLUMN;
×
5467
  }
5468
  if (!(aColData[0].flag == HAS_VALUE)) {
12,191!
5469
    return TSDB_CODE_PAR_PRIMARY_KEY_IS_NULL;
×
5470
  }
5471

5472
  if (aColData[0].nVal <= 1) goto _exit;
12,191✔
5473

5474
  int8_t doSort = 0;
12,056✔
5475
  int8_t doMerge = 0;
12,056✔
5476
  // scan -------
5477
  SRowKey lastKey;
5478
  tColDataArrGetRowKey(aColData, nColData, 0, &lastKey);
12,056✔
5479
  for (int32_t iVal = 1; iVal < aColData[0].nVal; ++iVal) {
10,118,300!
5480
    SRowKey key;
5481
    tColDataArrGetRowKey(aColData, nColData, iVal, &key);
10,155,323✔
5482

5483
    int32_t c = tRowKeyCompare(&lastKey, &key);
10,106,238✔
5484
    if (c < 0) {
10,106,238✔
5485
      lastKey = key;
10,083,103✔
5486
      continue;
10,083,103✔
5487
    } else if (c > 0) {
23,135✔
5488
      doSort = 1;
6✔
5489
      break;
6✔
5490
    } else {
5491
      doMerge = 1;
23,129✔
5492
    }
5493
  }
5494

5495
  // sort -------
5496
  if (doSort) {
×
5497
    TAOS_CHECK_RETURN(tColDataSort(aColData, nColData));
6!
5498
  }
5499

5500
  if ((doMerge != 1) && (doSort == 1)) {
12,084✔
5501
    tColDataArrGetRowKey(aColData, nColData, 0, &lastKey);
6✔
5502
    for (int32_t iVal = 1; iVal < aColData[0].nVal; ++iVal) {
28✔
5503
      SRowKey key;
5504
      tColDataArrGetRowKey(aColData, nColData, iVal, &key);
23✔
5505

5506
      int32_t c = tRowKeyCompare(&lastKey, &key);
23✔
5507
      if (c == 0) {
23✔
5508
        doMerge = 1;
1✔
5509
        break;
1✔
5510
      }
5511
      lastKey = key;
22✔
5512
    }
5513
  }
5514

5515
  // merge -------
5516
  if (doMerge) {
12,084✔
5517
    int32_t code = tColDataMerge(arr);
4✔
5518
    if (code) return code;
4!
5519
  }
5520

5521
_exit:
12,084✔
5522
  return 0;
12,219✔
5523
}
5524

5525
int32_t tColDataSortMergeWithBlob(SArray **arr, SBlobSet *pBlob) {
×
5526
  SArray   *colDataArr = *arr;
×
5527
  int32_t   nColData = TARRAY_SIZE(colDataArr);
×
5528
  SColData *aColData = (SColData *)TARRAY_DATA(colDataArr);
×
5529

5530
  if (!(aColData[0].type == TSDB_DATA_TYPE_TIMESTAMP)) {
×
5531
    return TSDB_CODE_PAR_INVALID_FIRST_COLUMN;
×
5532
  }
5533
  if (!(aColData[0].cid == PRIMARYKEY_TIMESTAMP_COL_ID)) {
×
5534
    return TSDB_CODE_PAR_INVALID_FIRST_COLUMN;
×
5535
  }
5536
  if (!(aColData[0].flag == HAS_VALUE)) {
×
5537
    return TSDB_CODE_PAR_PRIMARY_KEY_IS_NULL;
×
5538
  }
5539

5540
  if (aColData[0].nVal <= 1) goto _exit;
×
5541

5542
  int8_t doSort = 0;
×
5543
  int8_t doMerge = 0;
×
5544
  // scan -------
5545
  SRowKey lastKey;
5546
  tColDataArrGetRowKey(aColData, nColData, 0, &lastKey);
×
5547
  for (int32_t iVal = 1; iVal < aColData[0].nVal; ++iVal) {
×
5548
    SRowKey key;
5549
    tColDataArrGetRowKey(aColData, nColData, iVal, &key);
×
5550

5551
    int32_t c = tRowKeyCompare(&lastKey, &key);
×
5552
    if (c < 0) {
×
5553
      lastKey = key;
×
5554
      continue;
×
5555
    } else if (c > 0) {
×
5556
      doSort = 1;
×
5557
      break;
×
5558
    } else {
5559
      doMerge = 1;
×
5560
    }
5561
  }
5562
  if (doMerge || doSort) {
×
5563
    return TSDB_CODE_BLOB_NOT_SUPPORT;
×
5564
  }
5565
  // sort -------
5566
  if (doSort) {
×
5567
    TAOS_CHECK_RETURN(tColDataSort(aColData, nColData));
×
5568
  }
5569

5570
  if ((doMerge != 1) && (doSort == 1)) {
×
5571
    tColDataArrGetRowKey(aColData, nColData, 0, &lastKey);
×
5572
    for (int32_t iVal = 1; iVal < aColData[0].nVal; ++iVal) {
×
5573
      SRowKey key;
5574
      tColDataArrGetRowKey(aColData, nColData, iVal, &key);
×
5575

5576
      int32_t c = tRowKeyCompare(&lastKey, &key);
×
5577
      if (c == 0) {
×
5578
        doMerge = 1;
×
5579
        break;
×
5580
      }
5581
      lastKey = key;
×
5582
    }
5583
  }
5584

5585
  // merge -------
5586
  if (doMerge) {
×
5587
    int32_t code = tColDataMerge(arr);
×
5588
    if (code) return code;
×
5589
  }
5590

5591
_exit:
×
5592
  return 0;
×
5593
}
5594

5595
static int32_t tEncodeColDataVersion0(SEncoder *pEncoder, SColData *pColData) {
89,285✔
5596
  int32_t code = 0;
89,285✔
5597

5598
  if ((code = tEncodeI16v(pEncoder, pColData->cid))) return code;
178,570!
5599
  if ((code = tEncodeI8(pEncoder, pColData->type))) return code;
178,570!
5600
  if ((code = tEncodeI32v(pEncoder, pColData->nVal))) return code;
178,570!
5601
  if ((code = tEncodeI8(pEncoder, pColData->flag))) return code;
178,570!
5602

5603
  // bitmap
5604
  switch (pColData->flag) {
89,285!
5605
    case (HAS_NULL | HAS_NONE):
116✔
5606
    case (HAS_VALUE | HAS_NONE):
5607
    case (HAS_VALUE | HAS_NULL):
5608
      code = tEncodeFixed(pEncoder, pColData->pBitMap, BIT1_SIZE(pColData->nVal));
116✔
5609
      if (code) return code;
116!
5610
      break;
116✔
5611
    case (HAS_VALUE | HAS_NULL | HAS_NONE):
×
5612
      code = tEncodeFixed(pEncoder, pColData->pBitMap, BIT2_SIZE(pColData->nVal));
×
5613
      if (code) return code;
×
5614
      break;
×
5615
    default:
89,169✔
5616
      break;
89,169✔
5617
  }
5618

5619
  // value
5620
  if (pColData->flag & HAS_VALUE) {
89,285✔
5621
    if (IS_VAR_DATA_TYPE(pColData->type)) {
89,036!
5622
      code = tEncodeFixed(pEncoder, pColData->aOffset, pColData->nVal << 2);
1,862✔
5623
      if (code) return code;
1,862!
5624

5625
      code = tEncodeI32v(pEncoder, pColData->nData);
1,862✔
5626
      if (code) return code;
1,862!
5627

5628
      code = tEncodeFixed(pEncoder, pColData->pData, pColData->nData);
1,862✔
5629
      if (code) return code;
1,862!
5630
    } else {
5631
      code = tEncodeFixed(pEncoder, pColData->pData, pColData->nData);
87,174✔
5632
      if (code) return code;
87,174!
5633
    }
5634
  }
5635

5636
  return code;
89,285✔
5637
}
5638

5639
static int32_t tDecodeColDataVersion0(SDecoder *pDecoder, SColData *pColData) {
10,229✔
5640
  int32_t code = 0;
10,229✔
5641

5642
  if ((code = tDecodeI16v(pDecoder, &pColData->cid))) return code;
20,462!
5643
  if ((code = tDecodeI8(pDecoder, &pColData->type))) return code;
20,464!
5644
  if ((code = tDecodeI32v(pDecoder, &pColData->nVal))) return code;
20,459!
5645
  if ((code = tDecodeI8(pDecoder, &pColData->flag))) return code;
20,455!
5646

5647
  if (pColData->type <= 0 || pColData->type >= TSDB_DATA_TYPE_MAX || pColData->flag <= 0 || pColData->flag >= 8) {
10,227!
5648
    return TSDB_CODE_INVALID_PARA;
×
5649
  }
5650

5651
  // bitmap
5652
  switch (pColData->flag) {
10,229!
5653
    case (HAS_NULL | HAS_NONE):
234✔
5654
    case (HAS_VALUE | HAS_NONE):
5655
    case (HAS_VALUE | HAS_NULL):
5656
      code = tDecodeBinaryWithSize(pDecoder, BIT1_SIZE(pColData->nVal), &pColData->pBitMap);
234!
5657
      if (code) return code;
234!
5658
      break;
234✔
5659
    case (HAS_VALUE | HAS_NULL | HAS_NONE):
×
5660
      code = tDecodeBinaryWithSize(pDecoder, BIT2_SIZE(pColData->nVal), &pColData->pBitMap);
×
5661
      if (code) return code;
×
5662
      break;
×
5663
    default:
9,995✔
5664
      break;
9,995✔
5665
  }
5666

5667
  // value
5668
  if (pColData->flag & HAS_VALUE) {
10,229✔
5669
    if (IS_VAR_DATA_TYPE(pColData->type)) {
9,894!
5670
      code = tDecodeBinaryWithSize(pDecoder, pColData->nVal << 2, (uint8_t **)&pColData->aOffset);
1,609!
5671
      if (code) return code;
1,609!
5672

5673
      code = tDecodeI32v(pDecoder, &pColData->nData);
1,609✔
5674
      if (code) return code;
1,613!
5675

5676
      code = tDecodeBinaryWithSize(pDecoder, pColData->nData, &pColData->pData);
1,613!
5677
      if (code) return code;
1,613!
5678
    } else {
5679
      pColData->nData = TYPE_BYTES[pColData->type] * pColData->nVal;
8,285✔
5680
      code = tDecodeBinaryWithSize(pDecoder, pColData->nData, &pColData->pData);
8,285!
5681
      if (code) return code;
8,285!
5682
    }
5683
  }
5684
  pColData->cflag = 0;
10,233✔
5685

5686
  return code;
10,233✔
5687
}
5688

5689
static int32_t tEncodeColDataVersion1(SEncoder *pEncoder, SColData *pColData) {
89,294✔
5690
  int32_t code = tEncodeColDataVersion0(pEncoder, pColData);
89,294✔
5691
  if (code) return code;
89,295!
5692
  return tEncodeI8(pEncoder, pColData->cflag);
178,590✔
5693
}
5694

5695
static int32_t tDecodeColDataVersion1(SDecoder *pDecoder, SColData *pColData) {
10,229✔
5696
  int32_t code = tDecodeColDataVersion0(pDecoder, pColData);
10,229✔
5697
  if (code) return code;
10,229!
5698

5699
  code = tDecodeI8(pDecoder, &pColData->cflag);
10,229✔
5700
  return code;
10,228✔
5701
}
5702

5703
int32_t tEncodeColData(uint8_t version, SEncoder *pEncoder, SColData *pColData) {
89,247✔
5704
  if (version == 0) {
89,247!
5705
    return tEncodeColDataVersion0(pEncoder, pColData);
×
5706
  } else if (version == 1) {
89,247!
5707
    return tEncodeColDataVersion1(pEncoder, pColData);
89,294✔
5708
  } else {
5709
    return TSDB_CODE_INVALID_PARA;
×
5710
  }
5711
}
5712

5713
int32_t tDecodeColData(uint8_t version, SDecoder *pDecoder, SColData *pColData) {
10,227✔
5714
  if (version == 0) {
10,227!
5715
    return tDecodeColDataVersion0(pDecoder, pColData);
×
5716
  } else if (version == 1) {
10,227!
5717
    return tDecodeColDataVersion1(pDecoder, pColData);
10,229✔
5718
  } else {
5719
    return TSDB_CODE_INVALID_PARA;
×
5720
  }
5721
}
5722

5723
int32_t tEncodeRow(SEncoder *pEncoder, SRow *pRow) { return tEncodeFixed(pEncoder, pRow, pRow->len); }
2,147,483,647✔
5724

5725
int32_t tDecodeRow(SDecoder *pDecoder, SRow **ppRow) {
1,618,011,698✔
5726
  if (ppRow == NULL) {
1,618,011,698!
5727
    return TSDB_CODE_INVALID_PARA;
×
5728
  }
5729

5730
  if (pDecoder->pos + sizeof(SRow) > pDecoder->size) {
1,618,011,698!
5731
    return TSDB_CODE_OUT_OF_RANGE;
×
5732
  }
5733

5734
  SRow *pRow = (SRow *)(pDecoder->data + pDecoder->pos);
1,618,011,698✔
5735
  return tDecodeBinaryWithSize(pDecoder, pRow->len, (uint8_t **)ppRow);
2,147,483,647!
5736
}
5737

5738
int32_t tEncodeBlobSet(SEncoder *pEncoder, SBlobSet *pRow) {
×
5739
  int32_t code = 0;
×
5740
  int32_t lino = 0;
×
5741
  uint8_t compressed = 0;
×
5742
  int32_t compressSize = 0;
×
5743
  char   *p = NULL;
×
5744

5745
  TAOS_CHECK_EXIT(tEncodeI8(pEncoder, pRow->type));
×
5746
  TAOS_CHECK_EXIT(tEncodeI64(pEncoder, pRow->len));
×
5747

5748
  int32_t nSeq = taosArrayGetSize(pRow->pSeqTable);
×
5749
  TAOS_CHECK_EXIT(tEncodeI32(pEncoder, nSeq));
×
5750

5751
  // if (pRow->type) {
5752
  void *pIter = taosHashIterate(pRow->pSeqToffset, NULL);
×
5753
  while (pIter) {
×
5754
    uint64_t seq = *(uint64_t *)taosHashGetKey(pIter, NULL);
×
5755
    int32_t  idx = *(int32_t *)pIter;
×
5756
    TAOS_CHECK_EXIT(tEncodeU64(pEncoder, seq));
×
5757
    TAOS_CHECK_EXIT(tEncodeI32(pEncoder, idx));
×
5758

5759
    pIter = taosHashIterate(pRow->pSeqToffset, pIter);
×
5760
  }
5761
  //}
5762
  for (int32_t i = 0; i < nSeq; i++) {
×
5763
    SBlobValue *p = taosArrayGet(pRow->pSeqTable, i);
×
5764
    TAOS_CHECK_EXIT(tEncodeU64(pEncoder, p->offset));
×
5765
    TAOS_CHECK_EXIT(tEncodeU32(pEncoder, p->len));
×
5766
    TAOS_CHECK_EXIT(tEncodeU32(pEncoder, p->dataOffset));
×
5767
    TAOS_CHECK_EXIT(tEncodeI8(pEncoder, p->nextRow));
×
5768
    TAOS_CHECK_EXIT(tEncodeI8(pEncoder, p->type));
×
5769
  }
5770

5771
  TAOS_CHECK_EXIT(tEncodeFixed(pEncoder, pRow->data, pRow->len));
×
5772

5773
_exit:
×
5774
  return code;
×
5775
}
5776

5777
int32_t tDecodeBlobSet(SDecoder *pDecoder, SBlobSet **pBlobSet) {
×
5778
  int32_t    code = 0;
×
5779
  int32_t    lino = 0;
×
5780
  int32_t    nSeq = 0;
×
5781
  uint8_t    compressd = 0;
×
5782
  int32_t    compressSize = 0;
×
5783
  SBlobSet  *pBlob = taosMemCalloc(1, sizeof(SBlobSet));
×
5784
  if (pBlob == NULL) {
×
5785
    TAOS_CHECK_EXIT(terrno);
×
5786
  }
5787
  TAOS_CHECK_EXIT(tDecodeI8(pDecoder, &pBlob->type));
×
5788
  TAOS_CHECK_EXIT(tDecodeI64(pDecoder, &pBlob->len));
×
5789

5790
  TAOS_CHECK_EXIT(tDecodeI32(pDecoder, &nSeq));
×
5791

5792
  // if (pBlob->type) {
5793
  pBlob->pSeqToffset = taosHashInit(128, taosGetDefaultHashFunction(TSDB_DATA_TYPE_UBIGINT), false, HASH_NO_LOCK);
×
5794
  if (pBlob->pSeqToffset == NULL) {
×
5795
    TAOS_CHECK_EXIT(terrno);
×
5796
  }
5797
  for (int32_t i = 0; i < nSeq; i++) {
×
5798
    uint64_t seq;
5799
    int32_t  idx;
5800
    TAOS_CHECK_EXIT(tDecodeU64(pDecoder, &seq));
×
5801
    TAOS_CHECK_EXIT(tDecodeI32(pDecoder, &idx));
×
5802

5803
    code = taosHashPut(pBlob->pSeqToffset, &seq, sizeof(seq), &idx, sizeof(idx));
×
5804
    TAOS_CHECK_EXIT(code);
×
5805
  }
5806
  //}
5807

5808
  pBlob->pSeqTable = taosArrayInit(nSeq, sizeof(SBlobValue));
×
5809
  if (pBlob->pSeqTable == NULL) {
×
5810
    TAOS_CHECK_EXIT(terrno);
×
5811
  }
5812

5813
  for (int32_t i = 0; i < nSeq; i++) {
×
5814
    SBlobValue value;
5815
    TAOS_CHECK_EXIT(tDecodeU64(pDecoder, &value.offset));
×
5816
    TAOS_CHECK_EXIT(tDecodeU32(pDecoder, &value.len));
×
5817
    TAOS_CHECK_EXIT(tDecodeU32(pDecoder, &value.dataOffset));
×
5818
    TAOS_CHECK_EXIT(tDecodeI8(pDecoder, &value.nextRow));
×
5819
    TAOS_CHECK_EXIT(tDecodeI8(pDecoder, &value.type));
×
5820
    if (taosArrayPush(pBlob->pSeqTable, &value) == NULL) {
×
5821
      TAOS_CHECK_EXIT(terrno);
×
5822
    }
5823
  }
5824

5825
  pBlob->data = taosMemCalloc(1, pBlob->len);
×
5826
  if (pBlob->data == NULL) {
×
5827
    TAOS_CHECK_EXIT(terrno);
×
5828
  }
5829

5830
  TAOS_CHECK_EXIT(tDecodeFixed(pDecoder, pBlob->data, pBlob->len));
×
5831
  *pBlobSet = pBlob;
×
5832

5833
  uTrace("decode blob len:%d", (int32_t)(pBlob->len));
×
5834

5835
_exit:
×
5836
  if (code != 0) {
×
5837
    if (pBlob != NULL) {
×
5838
      taosMemoryFree(pBlob->data);
×
5839
      taosArrayDestroy(pBlob->pSeqTable);
×
5840
      taosMemoryFree(pBlob);
×
5841
    }
5842
  }
5843
  return code;
×
5844
}
5845

5846
#define CALC_SUM_MAX_MIN(SUM, MAX, MIN, VAL) \
5847
  do {                                       \
5848
    (SUM) += (VAL);                          \
5849
    if ((MAX) < (VAL)) (MAX) = (VAL);        \
5850
    if ((MIN) > (VAL)) (MIN) = (VAL);        \
5851
  } while (0)
5852

5853
static FORCE_INLINE void tColDataCalcSMABool(SColData *pColData, SColumnDataAgg *pAggs) {
11,322✔
5854
  int64_t *sum = &pAggs->sum, *max = &pAggs->max, *min = &pAggs->min;
11,322✔
5855
  int16_t *numOfNull = &pAggs->numOfNull;
11,322✔
5856
  *sum = 0;
11,322✔
5857
  *max = 0;
11,322✔
5858
  *min = 1;
11,322✔
5859
  *numOfNull = 0;
11,322✔
5860

5861
  int8_t val;
5862
  if (HAS_VALUE == pColData->flag) {
11,322✔
5863
    for (int32_t iVal = 0; iVal < pColData->nVal; iVal++) {
14,375,312✔
5864
      val = ((int8_t *)pColData->pData)[iVal] ? 1 : 0;
14,364,989✔
5865
      CALC_SUM_MAX_MIN(*sum, *max, *min, val);
14,364,989✔
5866
    }
5867
  } else {
5868
    for (int32_t iVal = 0; iVal < pColData->nVal; iVal++) {
2,845,372✔
5869
      switch (tColDataGetBitValue(pColData, iVal)) {
2,844,373!
5870
        case 0:
142,654✔
5871
        case 1:
5872
          (*numOfNull)++;
142,654✔
5873
          break;
142,654✔
5874
        case 2:
2,701,817✔
5875
          val = ((int8_t *)pColData->pData)[iVal] ? 1 : 0;
2,701,817✔
5876
          CALC_SUM_MAX_MIN(*sum, *max, *min, val);
2,701,817✔
5877
          break;
2,701,817✔
5878
        default:
×
5879
          break;
×
5880
      }
5881
    }
5882
  }
5883
}
11,322✔
5884

5885
static FORCE_INLINE void tColDataCalcSMATinyInt(SColData *pColData, SColumnDataAgg *pAggs) {
11,350✔
5886
  int64_t *sum = &pAggs->sum, *max = &pAggs->max, *min = &pAggs->min;
11,350✔
5887
  int16_t *numOfNull = &pAggs->numOfNull;
11,350✔
5888
  *sum = 0;
11,350✔
5889
  *max = INT8_MIN;
11,350✔
5890
  *min = INT8_MAX;
11,350✔
5891
  *numOfNull = 0;
11,350✔
5892

5893
  int8_t val;
5894
  if (HAS_VALUE == pColData->flag) {
11,350✔
5895
    for (int32_t iVal = 0; iVal < pColData->nVal; iVal++) {
14,430,412✔
5896
      val = ((int8_t *)pColData->pData)[iVal];
14,420,073✔
5897
      CALC_SUM_MAX_MIN(*sum, *max, *min, val);
14,420,073✔
5898
    }
5899
  } else {
5900
    for (int32_t iVal = 0; iVal < pColData->nVal; iVal++) {
2,857,635✔
5901
      switch (tColDataGetBitValue(pColData, iVal)) {
2,856,738!
5902
        case 0:
151,003✔
5903
        case 1:
5904
          (*numOfNull)++;
151,003✔
5905
          break;
151,003✔
5906
        case 2:
2,705,779✔
5907
          val = ((int8_t *)pColData->pData)[iVal];
2,705,779✔
5908
          CALC_SUM_MAX_MIN(*sum, *max, *min, val);
2,705,779✔
5909
          break;
2,705,779✔
5910
        default:
×
5911
          break;
×
5912
      }
5913
    }
5914
  }
5915
}
11,236✔
5916

5917
static FORCE_INLINE void tColDataCalcSMATinySmallInt(SColData *pColData, SColumnDataAgg *pAggs) {
11,339✔
5918
  int64_t *sum = &pAggs->sum, *max = &pAggs->max, *min = &pAggs->min;
11,339✔
5919
  int16_t *numOfNull = &pAggs->numOfNull;
11,339✔
5920
  *sum = 0;
11,339✔
5921
  *max = INT16_MIN;
11,339✔
5922
  *min = INT16_MAX;
11,339✔
5923
  *numOfNull = 0;
11,339✔
5924

5925
  int16_t val;
5926
  if (HAS_VALUE == pColData->flag) {
11,339✔
5927
    for (int32_t iVal = 0; iVal < pColData->nVal; iVal++) {
14,433,454✔
5928
      val = ((int16_t *)pColData->pData)[iVal];
14,423,114✔
5929
      CALC_SUM_MAX_MIN(*sum, *max, *min, val);
14,423,114✔
5930
    }
5931
  } else {
5932
    for (int32_t iVal = 0; iVal < pColData->nVal; iVal++) {
2,845,838✔
5933
      switch (tColDataGetBitValue(pColData, iVal)) {
2,844,839!
5934
        case 0:
142,793✔
5935
        case 1:
5936
          (*numOfNull)++;
142,793✔
5937
          break;
142,793✔
5938
        case 2:
2,702,134✔
5939
          val = ((int16_t *)pColData->pData)[iVal];
2,702,134✔
5940
          CALC_SUM_MAX_MIN(*sum, *max, *min, val);
2,702,134✔
5941
          break;
2,702,134✔
5942
        default:
×
5943
          break;
×
5944
      }
5945
    }
5946
  }
5947
}
11,339✔
5948

5949
static FORCE_INLINE void tColDataCalcSMAInt(SColData *pColData, SColumnDataAgg *pAggs) {
542,970✔
5950
  int64_t *sum = &pAggs->sum, *max = &pAggs->max, *min = &pAggs->min;
542,970✔
5951
  int16_t *numOfNull = &pAggs->numOfNull;
542,970✔
5952
  *sum = 0;
542,970✔
5953
  *max = INT32_MIN;
542,970✔
5954
  *min = INT32_MAX;
542,970✔
5955
  *numOfNull = 0;
542,970✔
5956

5957
  int32_t val;
5958
  if (HAS_VALUE == pColData->flag) {
542,970✔
5959
    for (int32_t iVal = 0; iVal < pColData->nVal; iVal++) {
1,708,810,299✔
5960
      val = ((int32_t *)pColData->pData)[iVal];
1,708,328,729✔
5961
      CALC_SUM_MAX_MIN(*sum, *max, *min, val);
1,708,328,729✔
5962
    }
5963
  } else {
5964
    for (int32_t iVal = 0; iVal < pColData->nVal; iVal++) {
164,072,236!
5965
      switch (tColDataGetBitValue(pColData, iVal)) {
164,092,641!
5966
        case 0:
9,107,767✔
5967
        case 1:
5968
          (*numOfNull)++;
9,107,767✔
5969
          break;
9,107,767✔
5970
        case 2:
155,071,486✔
5971
          val = ((int32_t *)pColData->pData)[iVal];
155,071,486✔
5972
          CALC_SUM_MAX_MIN(*sum, *max, *min, val);
155,071,486✔
5973
          break;
155,071,486✔
5974
        default:
×
5975
          break;
×
5976
      }
5977
    }
5978
  }
5979
}
461,165✔
5980

5981
static FORCE_INLINE void tColDataCalcSMABigInt(SColData *pColData, SColumnDataAgg *pAggs) {
271,909✔
5982
  int64_t *sum = &pAggs->sum, *max = &pAggs->max, *min = &pAggs->min;
271,909✔
5983
  int16_t *numOfNull = &pAggs->numOfNull;
271,909✔
5984
  *sum = 0;
271,909✔
5985
  *max = INT64_MIN;
271,909✔
5986
  *min = INT64_MAX;
271,909✔
5987
  *numOfNull = 0;
271,909✔
5988

5989
  int64_t val;
5990
  if (HAS_VALUE == pColData->flag) {
271,909✔
5991
    for (int32_t iVal = 0; iVal < pColData->nVal; iVal++) {
682,949,433✔
5992
      val = ((int64_t *)pColData->pData)[iVal];
682,683,219✔
5993
      CALC_SUM_MAX_MIN(*sum, *max, *min, val);
682,683,219✔
5994
    }
5995
  } else {
5996
    for (int32_t iVal = 0; iVal < pColData->nVal; iVal++) {
18,769,202✔
5997
      switch (tColDataGetBitValue(pColData, iVal)) {
18,763,509!
5998
        case 0:
1,797,396✔
5999
        case 1:
6000
          (*numOfNull)++;
1,797,396✔
6001
          break;
1,797,396✔
6002
        case 2:
16,966,302✔
6003
          val = ((int64_t *)pColData->pData)[iVal];
16,966,302✔
6004
          CALC_SUM_MAX_MIN(*sum, *max, *min, val);
16,966,302✔
6005
          break;
16,966,302✔
6006
        default:
×
6007
          break;
×
6008
      }
6009
    }
6010
  }
6011
}
271,907✔
6012

6013
static FORCE_INLINE void tColDataCalcSMAFloat(SColData *pColData, SColumnDataAgg *pAggs) {
484,388✔
6014
  int64_t *sum = &pAggs->sum, *max = &pAggs->max, *min = &pAggs->min;
484,388✔
6015
  int16_t *numOfNull = &pAggs->numOfNull;
484,388✔
6016
  *(double *)sum = 0;
484,388✔
6017
  *(double *)max = -FLT_MAX;
484,388✔
6018
  *(double *)min = FLT_MAX;
484,388✔
6019
  *numOfNull = 0;
484,388✔
6020

6021
  float val;
6022
  if (HAS_VALUE == pColData->flag) {
484,388✔
6023
    for (int32_t iVal = 0; iVal < pColData->nVal; iVal++) {
1,950,508,883✔
6024
      val = ((float *)pColData->pData)[iVal];
1,950,025,504✔
6025
      CALC_SUM_MAX_MIN(*(double *)sum, *(double *)max, *(double *)min, val);
1,950,025,504✔
6026
    }
6027
  } else {
6028
    for (int32_t iVal = 0; iVal < pColData->nVal; iVal++) {
2,857,381✔
6029
      switch (tColDataGetBitValue(pColData, iVal)) {
2,856,409!
6030
        case 0:
152,031✔
6031
        case 1:
6032
          (*numOfNull)++;
152,031✔
6033
          break;
152,031✔
6034
        case 2:
2,704,498✔
6035
          val = ((float *)pColData->pData)[iVal];
2,704,498✔
6036
          CALC_SUM_MAX_MIN(*(double *)sum, *(double *)max, *(double *)min, val);
2,704,498✔
6037
          break;
2,704,498✔
6038
        default:
×
6039
          break;
×
6040
      }
6041
    }
6042
  }
6043
}
484,351✔
6044

6045
static FORCE_INLINE void tColDataCalcSMADouble(SColData *pColData, SColumnDataAgg *pAggs) {
12,529✔
6046
  int64_t *sum = &pAggs->sum, *max = &pAggs->max, *min = &pAggs->min;
12,529✔
6047
  int16_t *numOfNull = &pAggs->numOfNull;
12,529✔
6048
  *(double *)sum = 0;
12,529✔
6049
  *(double *)max = -DBL_MAX;
12,529✔
6050
  *(double *)min = DBL_MAX;
12,529✔
6051
  *numOfNull = 0;
12,529✔
6052

6053
  double val;
6054
  if (HAS_VALUE == pColData->flag) {
12,529✔
6055
    for (int32_t iVal = 0; iVal < pColData->nVal; iVal++) {
21,273,029✔
6056
      val = ((double *)pColData->pData)[iVal];
21,261,511✔
6057
      CALC_SUM_MAX_MIN(*(double *)sum, *(double *)max, *(double *)min, val);
21,261,511✔
6058
    }
6059
  } else {
6060
    for (int32_t iVal = 0; iVal < pColData->nVal; iVal++) {
2,857,604✔
6061
      switch (tColDataGetBitValue(pColData, iVal)) {
2,856,588!
6062
        case 0:
151,559✔
6063
        case 1:
6064
          (*numOfNull)++;
151,559✔
6065
          break;
151,559✔
6066
        case 2:
2,705,170✔
6067
          val = ((double *)pColData->pData)[iVal];
2,705,170✔
6068
          CALC_SUM_MAX_MIN(*(double *)sum, *(double *)max, *(double *)min, val);
2,705,170✔
6069
          break;
2,705,170✔
6070
        default:
×
6071
          break;
×
6072
      }
6073
    }
6074
  }
6075
}
12,534✔
6076

6077
static FORCE_INLINE void tColDataCalcSMAUTinyInt(SColData *pColData, SColumnDataAgg *pAggs) {
8,557✔
6078
  int64_t *sum = &pAggs->sum, *max = &pAggs->max, *min = &pAggs->min;
8,557✔
6079
  int16_t *numOfNull = &pAggs->numOfNull;
8,557✔
6080
  *(uint64_t *)sum = 0;
8,557✔
6081
  *(uint64_t *)max = 0;
8,557✔
6082
  *(uint64_t *)min = UINT8_MAX;
8,557✔
6083
  *numOfNull = 0;
8,557✔
6084

6085
  uint8_t val;
6086
  if (HAS_VALUE == pColData->flag) {
8,557✔
6087
    for (int32_t iVal = 0; iVal < pColData->nVal; iVal++) {
13,995,773✔
6088
      val = ((uint8_t *)pColData->pData)[iVal];
13,988,215✔
6089
      CALC_SUM_MAX_MIN(*(uint64_t *)sum, *(uint64_t *)max, *(uint64_t *)min, val);
13,988,215✔
6090
    }
6091
  } else {
6092
    for (int32_t iVal = 0; iVal < pColData->nVal; iVal++) {
2,845,950✔
6093
      switch (tColDataGetBitValue(pColData, iVal)) {
2,844,978!
6094
        case 0:
142,122✔
6095
        case 1:
6096
          (*numOfNull)++;
142,122✔
6097
          break;
142,122✔
6098
        case 2:
2,702,926✔
6099
          val = ((uint8_t *)pColData->pData)[iVal];
2,702,926✔
6100
          CALC_SUM_MAX_MIN(*(uint64_t *)sum, *(uint64_t *)max, *(uint64_t *)min, val);
2,702,926✔
6101
          break;
2,702,926✔
6102
        default:
×
6103
          break;
×
6104
      }
6105
    }
6106
  }
6107
}
8,530✔
6108

6109
static FORCE_INLINE void tColDataCalcSMATinyUSmallInt(SColData *pColData, SColumnDataAgg *pAggs) {
8,556✔
6110
  int64_t *sum = &pAggs->sum, *max = &pAggs->max, *min = &pAggs->min;
8,556✔
6111
  int16_t *numOfNull = &pAggs->numOfNull;
8,556✔
6112
  *(uint64_t *)sum = 0;
8,556✔
6113
  *(uint64_t *)max = 0;
8,556✔
6114
  *(uint64_t *)min = UINT16_MAX;
8,556✔
6115
  *numOfNull = 0;
8,556✔
6116

6117
  uint16_t val;
6118
  if (HAS_VALUE == pColData->flag) {
8,556✔
6119
    for (int32_t iVal = 0; iVal < pColData->nVal; iVal++) {
13,995,383✔
6120
      val = ((uint16_t *)pColData->pData)[iVal];
13,987,825✔
6121
      CALC_SUM_MAX_MIN(*(uint64_t *)sum, *(uint64_t *)max, *(uint64_t *)min, val);
13,987,825✔
6122
    }
6123
  } else {
6124
    for (int32_t iVal = 0; iVal < pColData->nVal; iVal++) {
2,846,351✔
6125
      switch (tColDataGetBitValue(pColData, iVal)) {
2,845,353!
6126
        case 0:
142,518✔
6127
        case 1:
6128
          (*numOfNull)++;
142,518✔
6129
          break;
142,518✔
6130
        case 2:
2,702,900✔
6131
          val = ((uint16_t *)pColData->pData)[iVal];
2,702,900✔
6132
          CALC_SUM_MAX_MIN(*(uint64_t *)sum, *(uint64_t *)max, *(uint64_t *)min, val);
2,702,900✔
6133
          break;
2,702,900✔
6134
        default:
×
6135
          break;
×
6136
      }
6137
    }
6138
  }
6139
}
8,556✔
6140

6141
static FORCE_INLINE void tColDataCalcSMAUInt(SColData *pColData, SColumnDataAgg *pAggs) {
8,556✔
6142
  int64_t *sum = &pAggs->sum, *max = &pAggs->max, *min = &pAggs->min;
8,556✔
6143
  int16_t *numOfNull = &pAggs->numOfNull;
8,556✔
6144
  *(uint64_t *)sum = 0;
8,556✔
6145
  *(uint64_t *)max = 0;
8,556✔
6146
  *(uint64_t *)min = UINT32_MAX;
8,556✔
6147
  *numOfNull = 0;
8,556✔
6148

6149
  uint32_t val;
6150
  if (HAS_VALUE == pColData->flag) {
8,556✔
6151
    for (int32_t iVal = 0; iVal < pColData->nVal; iVal++) {
13,995,202✔
6152
      val = ((uint32_t *)pColData->pData)[iVal];
13,987,644✔
6153
      CALC_SUM_MAX_MIN(*(uint64_t *)sum, *(uint64_t *)max, *(uint64_t *)min, val);
13,987,644✔
6154
    }
6155
  } else {
6156
    for (int32_t iVal = 0; iVal < pColData->nVal; iVal++) {
2,846,250✔
6157
      switch (tColDataGetBitValue(pColData, iVal)) {
2,845,230!
6158
        case 0:
142,008✔
6159
        case 1:
6160
          (*numOfNull)++;
142,008✔
6161
          break;
142,008✔
6162
        case 2:
2,703,296✔
6163
          val = ((uint32_t *)pColData->pData)[iVal];
2,703,296✔
6164
          CALC_SUM_MAX_MIN(*(uint64_t *)sum, *(uint64_t *)max, *(uint64_t *)min, val);
2,703,296✔
6165
          break;
2,703,296✔
6166
        default:
×
6167
          break;
×
6168
      }
6169
    }
6170
  }
6171
}
8,578✔
6172

6173
static FORCE_INLINE void tColDataCalcSMAUBigInt(SColData *pColData, SColumnDataAgg *pAggs) {
8,557✔
6174
  int64_t *sum = &pAggs->sum, *max = &pAggs->max, *min = &pAggs->min;
8,557✔
6175
  int16_t *numOfNull = &pAggs->numOfNull;
8,557✔
6176
  *(uint64_t *)sum = 0;
8,557✔
6177
  *(uint64_t *)max = 0;
8,557✔
6178
  *(uint64_t *)min = UINT64_MAX;
8,557✔
6179
  *numOfNull = 0;
8,557✔
6180

6181
  uint64_t val;
6182
  if (HAS_VALUE == pColData->flag) {
8,557✔
6183
    for (int32_t iVal = 0; iVal < pColData->nVal; iVal++) {
13,995,054✔
6184
      val = ((uint64_t *)pColData->pData)[iVal];
13,987,496✔
6185
      CALC_SUM_MAX_MIN(*(uint64_t *)sum, *(uint64_t *)max, *(uint64_t *)min, val);
13,987,496✔
6186
    }
6187
  } else {
6188
    for (int32_t iVal = 0; iVal < pColData->nVal; iVal++) {
2,846,427✔
6189
      switch (tColDataGetBitValue(pColData, iVal)) {
2,845,350✔
6190
        case 0:
142,797✔
6191
        case 1:
6192
          (*numOfNull)++;
142,797✔
6193
          break;
142,797✔
6194
        case 2:
2,702,627✔
6195
          val = ((uint64_t *)pColData->pData)[iVal];
2,702,627✔
6196
          CALC_SUM_MAX_MIN(*(uint64_t *)sum, *(uint64_t *)max, *(uint64_t *)min, val);
2,702,627✔
6197
          break;
2,702,627✔
6198
        default:
4✔
6199
          break;
4✔
6200
      }
6201
    }
6202
  }
6203
}
8,635✔
6204

6205
static FORCE_INLINE void tColDataCalcSMAVarType(SColData *pColData, SColumnDataAgg *pAggs) {
28,299✔
6206
  int64_t *sum = &pAggs->sum, *max = &pAggs->max, *min = &pAggs->min;
28,299✔
6207
  int16_t *numOfNull = &pAggs->numOfNull;
28,299✔
6208
  *(uint64_t *)sum = 0;
28,299✔
6209
  *(uint64_t *)max = 0;
28,299✔
6210
  *(uint64_t *)min = 0;
28,299✔
6211
  *numOfNull = 0;
28,299✔
6212

6213
  switch (pColData->flag) {
28,299!
6214
    case HAS_NONE:
×
6215
    case HAS_NULL:
6216
    case (HAS_NONE | HAS_NULL):
6217
      *numOfNull = pColData->nVal;
×
6218
      break;
×
6219
    case HAS_VALUE:
26,290✔
6220
      *numOfNull = 0;
26,290✔
6221
      break;
26,290✔
6222
    case (HAS_VALUE | HAS_NULL):
2,010✔
6223
    case (HAS_VALUE | HAS_NONE):
6224
      for (int32_t iVal = 0; iVal < pColData->nVal; iVal++) {
5,705,257✔
6225
        if (GET_BIT1(pColData->pBitMap, iVal) == 0) {
5,703,247✔
6226
          (*numOfNull)++;
293,932✔
6227
        }
6228
      }
6229
      break;
2,010✔
6230
    case (HAS_VALUE | HAS_NONE | HAS_NULL):
×
6231
      for (int32_t iVal = 0; iVal < pColData->nVal; iVal++) {
×
6232
        if (GET_BIT2(pColData->pBitMap, iVal) != 2) {
×
6233
          (*numOfNull)++;
×
6234
        }
6235
      }
6236
      break;
×
6237
    default:
×
6238
      break;
×
6239
  }
6240
}
28,299✔
6241

6242
#define CALC_DECIMAL_SUM_MAX_MIN(TYPE, pSumOp, pCompOp, pColData, pSum, pMax, pMin)                   \
6243
  do {                                                                                                \
6244
    if (decimal128AddCheckOverflow((Decimal *)pSum, pVal, DECIMAL_WORD_NUM(TYPE))) *pOverflow = true; \
6245
    pSumOp->add(pSum, pVal, DECIMAL_WORD_NUM(TYPE));                                                  \
6246
    if (pCompOp->gt(pVal, pMax, DECIMAL_WORD_NUM(TYPE))) {                                            \
6247
      *(pMax) = *pVal;                                                                                \
6248
    }                                                                                                 \
6249
    if (pCompOp->lt(pVal, pMin, DECIMAL_WORD_NUM(TYPE))) {                                            \
6250
      *(pMin) = *pVal;                                                                                \
6251
    }                                                                                                 \
6252
  } while (0)
6253

6254
static FORCE_INLINE void tColDataCalcSMADecimal64Type(SColData *pColData, SColumnDataAgg *pAggs) {
723✔
6255
  Decimal128 *pSum = (Decimal128 *)pAggs->decimal128Sum;
723✔
6256
  Decimal64  *pMax = (Decimal64 *)pAggs->decimal128Max, *pMin = (Decimal64 *)pAggs->decimal128Min;
723✔
6257
  uint8_t    *pOverflow = &pAggs->overflow;
723✔
6258
  *pSum = DECIMAL128_ZERO;
723✔
6259
  *pMax = DECIMAL64_MIN;
723✔
6260
  *pMin = DECIMAL64_MAX;
723✔
6261
  pAggs->numOfNull = 0;
723✔
6262
  pAggs->colId |= DECIMAL_AGG_FLAG;
723✔
6263

6264
  Decimal64         *pVal = NULL;
723✔
6265
  const SDecimalOps *pSumOps = getDecimalOps(TSDB_DATA_TYPE_DECIMAL);
723✔
6266
  const SDecimalOps *pCompOps = getDecimalOps(TSDB_DATA_TYPE_DECIMAL64);
723✔
6267
  if (HAS_VALUE == pColData->flag) {
723✔
6268
    for (int32_t iVal = 0; iVal < pColData->nVal; ++iVal) {
2,315,972✔
6269
      pVal = ((Decimal64 *)pColData->pData) + iVal;
2,315,273✔
6270
      CALC_DECIMAL_SUM_MAX_MIN(Decimal64, pSumOps, pCompOps, pColData, pSum, pMax, pMin);
2,315,273!
6271
    }
6272
  } else {
6273
    for (int32_t iVal = 0; iVal < pColData->nVal; ++iVal) {
12,012✔
6274
      switch (tColDataGetBitValue(pColData, iVal)) {
12,000!
6275
        case 0:
3,826✔
6276
        case 1:
6277
          pAggs->numOfNull++;
3,826✔
6278
          break;
3,826✔
6279
        case 2:
8,174✔
6280
          pVal = ((Decimal64 *)pColData->pData) + iVal;
8,174✔
6281
          CALC_DECIMAL_SUM_MAX_MIN(Decimal64, pSumOps, pCompOps, pColData, pSum, pMax, pMin);
8,174!
6282
          break;
8,174✔
6283
        default:
×
6284
          break;
×
6285
      }
6286
    }
6287
  }
6288
}
711✔
6289

6290
static FORCE_INLINE void tColDataCalcSMADecimal128Type(SColData *pColData, SColumnDataAgg *pAggs) {
747✔
6291
  Decimal128 *pSum = (Decimal128 *)pAggs->decimal128Sum, *pMax = (Decimal128 *)pAggs->decimal128Max,
747✔
6292
             *pMin = (Decimal128 *)pAggs->decimal128Min;
747✔
6293
  uint8_t *pOverflow = &pAggs->overflow;
747✔
6294
  *pSum = DECIMAL128_ZERO;
747✔
6295
  *pMax = DECIMAL128_MIN;
747✔
6296
  *pMin = DECIMAL128_MAX;
747✔
6297
  pAggs->numOfNull = 0;
747✔
6298
  pAggs->colId |= DECIMAL_AGG_FLAG;
747✔
6299

6300
  Decimal128        *pVal = NULL;
747✔
6301
  const SDecimalOps *pOps = getDecimalOps(TSDB_DATA_TYPE_DECIMAL);
747✔
6302
  if (HAS_VALUE == pColData->flag) {
747✔
6303
    for (int32_t iVal = 0; iVal < pColData->nVal; ++iVal) {
2,319,201✔
6304
      pVal = ((Decimal128 *)pColData->pData) + iVal;
2,318,433✔
6305
      CALC_DECIMAL_SUM_MAX_MIN(Decimal128, pOps, pOps, pColData, pSum, pMax, pMin);
2,318,433!
6306
    }
6307
  } else {
6308
    for (int32_t iVal = 0; iVal < pColData->nVal; ++iVal) {
36,036✔
6309
      switch (tColDataGetBitValue(pColData, iVal)) {
36,000!
6310
        case 0:
11,504✔
6311
        case 1:
6312
          pAggs->numOfNull++;
11,504✔
6313
          break;
11,504✔
6314
        case 2:
24,496✔
6315
          pVal = ((Decimal128 *)pColData->pData) + iVal;
24,496✔
6316
          CALC_DECIMAL_SUM_MAX_MIN(Decimal128, pOps, pOps, pColData, pSum, pMax, pMin);
24,496✔
6317
          break;
24,496✔
6318
        default:
×
6319
          break;
×
6320
      }
6321
    }
6322
  }
6323
}
804✔
6324

6325
void (*tColDataCalcSMA[])(SColData *pColData, SColumnDataAgg *pAggs) = {
6326
    NULL,
6327
    tColDataCalcSMABool,            // TSDB_DATA_TYPE_BOOL
6328
    tColDataCalcSMATinyInt,         // TSDB_DATA_TYPE_TINYINT
6329
    tColDataCalcSMATinySmallInt,    // TSDB_DATA_TYPE_SMALLINT
6330
    tColDataCalcSMAInt,             // TSDB_DATA_TYPE_INT
6331
    tColDataCalcSMABigInt,          // TSDB_DATA_TYPE_BIGINT
6332
    tColDataCalcSMAFloat,           // TSDB_DATA_TYPE_FLOAT
6333
    tColDataCalcSMADouble,          // TSDB_DATA_TYPE_DOUBLE
6334
    tColDataCalcSMAVarType,         // TSDB_DATA_TYPE_VARCHAR
6335
    tColDataCalcSMABigInt,          // TSDB_DATA_TYPE_TIMESTAMP
6336
    tColDataCalcSMAVarType,         // TSDB_DATA_TYPE_NCHAR
6337
    tColDataCalcSMAUTinyInt,        // TSDB_DATA_TYPE_UTINYINT
6338
    tColDataCalcSMATinyUSmallInt,   // TSDB_DATA_TYPE_USMALLINT
6339
    tColDataCalcSMAUInt,            // TSDB_DATA_TYPE_UINT
6340
    tColDataCalcSMAUBigInt,         // TSDB_DATA_TYPE_UBIGINT
6341
    tColDataCalcSMAVarType,         // TSDB_DATA_TYPE_JSON
6342
    tColDataCalcSMAVarType,         // TSDB_DATA_TYPE_VARBINARY
6343
    tColDataCalcSMADecimal128Type,  // TSDB_DATA_TYPE_DECIMAL
6344
    tColDataCalcSMAVarType,         // TSDB_DATA_TYPE_BLOB
6345
    NULL,                           // TSDB_DATA_TYPE_MEDIUMBLOB
6346
    tColDataCalcSMAVarType,         // TSDB_DATA_TYPE_GEOMETRY
6347
    tColDataCalcSMADecimal64Type,   // TSDB_DATA_TYPE_DECIMAL64
6348
};
6349

6350
// SValueColumn ================================
6351
int32_t tValueColumnInit(SValueColumn *valCol) {
32,715,753✔
6352
  valCol->type = TSDB_DATA_TYPE_NULL;
32,715,753✔
6353
  valCol->numOfValues = 0;
32,715,753✔
6354
  tBufferInit(&valCol->data);
32,715,753✔
6355
  tBufferInit(&valCol->offsets);
32,715,753✔
6356
  return 0;
32,715,753✔
6357
}
6358

6359
void tValueColumnDestroy(SValueColumn *valCol) {
74,335,242✔
6360
  valCol->type = TSDB_DATA_TYPE_NULL;
74,335,242✔
6361
  valCol->numOfValues = 0;
74,335,242✔
6362
  tBufferDestroy(&valCol->data);
74,335,242✔
6363
  tBufferDestroy(&valCol->offsets);
74,335,382✔
6364
  return;
74,335,372✔
6365
}
6366

6367
void tValueColumnClear(SValueColumn *valCol) {
35,979,512✔
6368
  valCol->type = TSDB_DATA_TYPE_NULL;
35,979,512✔
6369
  valCol->numOfValues = 0;
35,979,512✔
6370
  tBufferClear(&valCol->data);
35,979,512✔
6371
  tBufferClear(&valCol->offsets);
35,979,512✔
6372
  return;
35,979,512✔
6373
}
6374

6375
int32_t tValueColumnAppend(SValueColumn *valCol, const SValue *value) {
1,817,009✔
6376
  int32_t code;
6377

6378
  if (valCol->numOfValues == 0) {
1,817,009✔
6379
    valCol->type = value->type;
15,045✔
6380
  }
6381

6382
  if (!(value->type == valCol->type)) {
1,817,009!
6383
    return TSDB_CODE_INVALID_PARA;
×
6384
  }
6385

6386
  if (IS_VAR_DATA_TYPE(value->type)) {
1,817,009!
6387
    if ((code = tBufferPutI32(&valCol->offsets, tBufferGetSize(&valCol->data)))) {
12,485!
6388
      return code;
×
6389
    }
6390
    if ((code = tBufferPut(&valCol->data, value->pData, value->nData))) {
12,486!
6391
      return code;
×
6392
    }
6393
  } else {
6394
    code = tBufferPut(&valCol->data, VALUE_GET_DATUM(value, value->type), tDataTypes[value->type].bytes);
1,810,767!
6395
    if (code) return code;
1,810,767!
6396
  }
6397
  valCol->numOfValues++;
1,817,010✔
6398

6399
  return 0;
1,817,010✔
6400
}
6401

6402
int32_t tValueColumnUpdate(SValueColumn *valCol, int32_t idx, const SValue *value) {
468,386,047✔
6403
  int32_t code;
6404

6405
  if (idx < 0 || idx >= valCol->numOfValues) {
468,386,047!
6406
    return TSDB_CODE_OUT_OF_RANGE;
×
6407
  }
6408

6409
  if (IS_VAR_DATA_TYPE(valCol->type)) {
468,439,634!
6410
    int32_t *offsets = (int32_t *)tBufferGetData(&valCol->offsets);
×
6411
    int32_t  nextOffset = (idx == valCol->numOfValues - 1) ? tBufferGetSize(&valCol->data) : offsets[idx + 1];
×
6412
    int32_t  oldDataSize = nextOffset - offsets[idx];
×
6413
    int32_t  bytesAdded = value->nData - oldDataSize;
×
6414

6415
    if (bytesAdded != 0) {
×
6416
      if ((code = tBufferEnsureCapacity(&valCol->data, tBufferGetSize(&valCol->data) + bytesAdded))) return code;
72!
6417
      memmove(tBufferGetDataAt(&valCol->data, nextOffset + bytesAdded), tBufferGetDataAt(&valCol->data, nextOffset),
36✔
6418
              tBufferGetSize(&valCol->data) - nextOffset);
36✔
6419
      valCol->data.size += bytesAdded;
36✔
6420

6421
      for (int32_t i = idx + 1; i < valCol->numOfValues; i++) {
36!
6422
        offsets[i] += bytesAdded;
×
6423
      }
6424
    }
6425
    return tBufferPutAt(&valCol->data, offsets[idx], value->pData, value->nData);
×
6426
  } else {
6427
    return tBufferPutAt(&valCol->data, idx * tDataTypes[valCol->type].bytes, VALUE_GET_DATUM(value, valCol->type),
468,521,137!
6428
                        tDataTypes[valCol->type].bytes);
468,521,137✔
6429
  }
6430
  return 0;
6431
}
6432

6433
int32_t tValueColumnGet(SValueColumn *valCol, int32_t idx, SValue *value) {
945,516,469✔
6434
  if (idx < 0 || idx >= valCol->numOfValues) {
945,516,469!
6435
    return TSDB_CODE_OUT_OF_RANGE;
×
6436
  }
6437

6438
  value->type = valCol->type;
945,643,085✔
6439
  if (IS_VAR_DATA_TYPE(value->type)) {
945,643,085!
6440
    int32_t       offset, nextOffset;
6441
    SBufferReader reader = BUFFER_READER_INITIALIZER(idx * sizeof(offset), &valCol->offsets);
×
6442

6443
    TAOS_CHECK_RETURN(tBufferGetI32(&reader, &offset));
×
6444
    if (idx == valCol->numOfValues - 1) {
339,001✔
6445
      nextOffset = tBufferGetSize(&valCol->data);
156,931✔
6446
    } else {
6447
      TAOS_CHECK_RETURN(tBufferGetI32(&reader, &nextOffset));
182,070✔
6448
    }
6449
    value->nData = nextOffset - offset;
338,753✔
6450
    value->pData = (uint8_t *)tBufferGetDataAt(&valCol->data, offset);
338,753✔
6451
  } else {
6452
    SBufferReader reader = BUFFER_READER_INITIALIZER(idx * tDataTypes[value->type].bytes, &valCol->data);
945,643,226✔
6453
    TAOS_CHECK_RETURN(tBufferGet(&reader, tDataTypes[value->type].bytes, VALUE_GET_DATUM(value, value->type)));
1,891,286,452!
6454
  }
6455
  return 0;
945,981,979✔
6456
}
6457

6458
int32_t tValueColumnCompress(SValueColumn *valCol, SValueColumnCompressInfo *info, SBuffer *output, SBuffer *assist) {
15,043✔
6459
  int32_t code;
6460

6461
  if (!(valCol->numOfValues > 0)) {
15,043!
6462
    return TSDB_CODE_INVALID_PARA;
×
6463
  }
6464

6465
  (*info) = (SValueColumnCompressInfo){
15,043✔
6466
      .cmprAlg = info->cmprAlg,
15,043✔
6467
      .type = valCol->type,
15,043✔
6468
  };
6469

6470
  // offset
6471
  if (IS_VAR_DATA_TYPE(valCol->type)) {
15,043!
6472
    SCompressInfo cinfo = {
2,420✔
6473
        .cmprAlg = info->cmprAlg,
2,420✔
6474
        .dataType = TSDB_DATA_TYPE_INT,
6475
        .originalSize = valCol->offsets.size,
2,420✔
6476
    };
6477

6478
    code = tCompressDataToBuffer(valCol->offsets.data, &cinfo, output, assist);
2,420✔
6479
    if (code) return code;
2,420!
6480

6481
    info->offsetOriginalSize = cinfo.originalSize;
2,420✔
6482
    info->offsetCompressedSize = cinfo.compressedSize;
2,420✔
6483
  }
6484

6485
  // data
6486
  SCompressInfo cinfo = {
15,043✔
6487
      .cmprAlg = info->cmprAlg,
15,043✔
6488
      .dataType = valCol->type,
15,043✔
6489
      .originalSize = valCol->data.size,
15,043✔
6490
  };
6491

6492
  code = tCompressDataToBuffer(valCol->data.data, &cinfo, output, assist);
15,043✔
6493
  if (code) return code;
15,045!
6494

6495
  info->dataOriginalSize = cinfo.originalSize;
15,045✔
6496
  info->dataCompressedSize = cinfo.compressedSize;
15,045✔
6497

6498
  return 0;
15,045✔
6499
}
6500

6501
int32_t tValueColumnDecompress(void *input, const SValueColumnCompressInfo *info, SValueColumn *valCol,
498,518✔
6502
                               SBuffer *assist) {
6503
  int32_t code;
6504

6505
  tValueColumnClear(valCol);
498,518✔
6506
  valCol->type = info->type;
498,564✔
6507
  // offset
6508
  if (IS_VAR_DATA_TYPE(valCol->type)) {
650,433!
6509
    valCol->numOfValues = info->offsetOriginalSize / tDataTypes[TSDB_DATA_TYPE_INT].bytes;
151,748✔
6510

6511
    SCompressInfo cinfo = {
151,748✔
6512
        .dataType = TSDB_DATA_TYPE_INT,
6513
        .cmprAlg = info->cmprAlg,
151,748✔
6514
        .originalSize = info->offsetOriginalSize,
151,748✔
6515
        .compressedSize = info->offsetCompressedSize,
151,748✔
6516
    };
6517

6518
    code = tDecompressDataToBuffer(input, &cinfo, &valCol->offsets, assist);
151,748✔
6519
    if (code) {
151,869!
6520
      return code;
×
6521
    }
6522
  } else {
6523
    valCol->numOfValues = info->dataOriginalSize / tDataTypes[valCol->type].bytes;
346,816✔
6524
  }
6525

6526
  // data
6527
  SCompressInfo cinfo = {
498,685✔
6528
      .dataType = valCol->type,
498,685✔
6529
      .cmprAlg = info->cmprAlg,
498,685✔
6530
      .originalSize = info->dataOriginalSize,
498,685✔
6531
      .compressedSize = info->dataCompressedSize,
498,685✔
6532
  };
6533

6534
  code = tDecompressDataToBuffer((char *)input + info->offsetCompressedSize, &cinfo, &valCol->data, assist);
498,685✔
6535
  if (code) {
498,744!
6536
    return code;
×
6537
  }
6538

6539
  return 0;
498,744✔
6540
}
6541

6542
int32_t tValueColumnCompressInfoEncode(const SValueColumnCompressInfo *info, SBuffer *buffer) {
15,045✔
6543
  int32_t code;
6544
  uint8_t fmtVer = 0;
15,045✔
6545

6546
  if ((code = tBufferPutU8(buffer, fmtVer))) return code;
30,090!
6547
  if ((code = tBufferPutI8(buffer, info->cmprAlg))) return code;
30,090!
6548
  if ((code = tBufferPutI8(buffer, info->type))) return code;
30,090!
6549
  if (IS_VAR_DATA_TYPE(info->type)) {
15,045!
6550
    if ((code = tBufferPutI32v(buffer, info->offsetOriginalSize))) return code;
4,838!
6551
    if ((code = tBufferPutI32v(buffer, info->offsetCompressedSize))) return code;
4,838!
6552
  }
6553
  if ((code = tBufferPutI32v(buffer, info->dataOriginalSize))) return code;
30,090!
6554
  if ((code = tBufferPutI32v(buffer, info->dataCompressedSize))) return code;
30,090!
6555

6556
  return 0;
15,045✔
6557
}
6558

6559
int32_t tValueColumnCompressInfoDecode(SBufferReader *reader, SValueColumnCompressInfo *info) {
498,534✔
6560
  int32_t code;
6561
  uint8_t fmtVer;
6562

6563
  if ((code = tBufferGetU8(reader, &fmtVer))) return code;
498,534!
6564
  if (fmtVer == 0) {
498,548!
6565
    if ((code = tBufferGetI8(reader, &info->cmprAlg))) return code;
498,562!
6566
    if ((code = tBufferGetI8(reader, &info->type))) return code;
498,496!
6567
    if (IS_VAR_DATA_TYPE(info->type)) {
498,473!
6568
      if ((code = tBufferGetI32v(reader, &info->offsetOriginalSize))) return code;
151,699!
6569
      if ((code = tBufferGetI32v(reader, &info->offsetCompressedSize))) return code;
151,773!
6570
    } else {
6571
      info->offsetOriginalSize = 0;
346,774✔
6572
      info->offsetCompressedSize = 0;
346,774✔
6573
    }
6574
    if ((code = tBufferGetI32v(reader, &info->dataOriginalSize))) return code;
498,510!
6575
    if ((code = tBufferGetI32v(reader, &info->dataCompressedSize))) return code;
498,480!
6576
  } else {
6577
    return TSDB_CODE_INVALID_PARA;
×
6578
  }
6579

6580
  return 0;
498,490✔
6581
}
6582

6583
int32_t tCompressData(void          *input,       // input
7,796,143✔
6584
                      SCompressInfo *info,        // compress info
6585
                      void          *output,      // output
6586
                      int32_t        outputSize,  // output size
6587
                      SBuffer       *buffer       // assistant buffer provided by caller, can be NULL
6588
) {
6589
  int32_t extraSizeNeeded;
6590
  int32_t code;
6591

6592
  extraSizeNeeded = (info->cmprAlg == NO_COMPRESSION) ? info->originalSize : info->originalSize + COMP_OVERFLOW_BYTES;
7,796,143✔
6593
  if (!(outputSize >= extraSizeNeeded)) {
7,796,143!
6594
    return TSDB_CODE_INVALID_PARA;
×
6595
  }
6596

6597
  if (info->cmprAlg == NO_COMPRESSION) {
7,796,143✔
6598
    (void)memcpy(output, input, info->originalSize);
244✔
6599
    info->compressedSize = info->originalSize;
244✔
6600
  } else if (info->cmprAlg == ONE_STAGE_COMP || info->cmprAlg == TWO_STAGE_COMP) {
9,902,141✔
6601
    SBuffer local;
6602

6603
    tBufferInit(&local);
6604
    if (buffer == NULL) {
2,105,890!
6605
      buffer = &local;
×
6606
    }
6607

6608
    if (info->cmprAlg == TWO_STAGE_COMP) {
2,105,890!
6609
      code = tBufferEnsureCapacity(buffer, extraSizeNeeded);
2,106,019✔
6610
      if (code) {
2,106,019!
6611
        tBufferDestroy(&local);
6612
        return code;
×
6613
      }
6614
    }
6615

6616
    info->compressedSize = tDataTypes[info->dataType].compFunc(  //
4,212,132✔
6617
        input,                                                   // input
6618
        info->originalSize,                                      // input size
6619
        info->originalSize / tDataTypes[info->dataType].bytes,   // number of elements
2,105,890✔
6620
        output,                                                  // output
6621
        outputSize,                                              // output size
6622
        info->cmprAlg,                                           // compression algorithm
2,105,890✔
6623
        buffer->data,                                            // buffer
6624
        buffer->capacity                                         // buffer size
2,105,890✔
6625
    );
6626
    if (info->compressedSize < 0) {
2,106,242!
6627
      tBufferDestroy(&local);
6628
      return TSDB_CODE_COMPRESS_ERROR;
×
6629
    }
6630

6631
    tBufferDestroy(&local);
6632
  } else {
6633
    DEFINE_VAR(info->cmprAlg)
5,690,009✔
6634
    if ((l1 == L1_UNKNOWN && l2 == L2_UNKNOWN) || (l1 == L1_DISABLED && l2 == L2_DISABLED)) {
5,690,009!
6635
      (void)memcpy(output, input, info->originalSize);
12✔
6636
      info->compressedSize = info->originalSize;
12✔
6637
      return 0;
12✔
6638
    }
6639
    SBuffer local;
6640

6641
    tBufferInit(&local);
6642
    if (buffer == NULL) {
5,689,997!
6643
      buffer = &local;
×
6644
    }
6645
    code = tBufferEnsureCapacity(buffer, extraSizeNeeded);
5,689,997✔
6646

6647
    info->compressedSize = tDataCompress[info->dataType].compFunc(  //
11,380,425✔
6648
        input,                                                      // input
6649
        info->originalSize,                                         // input size
6650
        info->originalSize / tDataTypes[info->dataType].bytes,      // number of elements
5,690,005✔
6651
        output,                                                     // output
6652
        outputSize,                                                 // output size
6653
        info->cmprAlg,                                              // compression algorithm
6654
        buffer->data,                                               // buffer
6655
        buffer->capacity                                            // buffer size
5,690,005✔
6656
    );
6657
    if (info->compressedSize < 0) {
5,690,420!
6658
      tBufferDestroy(&local);
6659
      return TSDB_CODE_COMPRESS_ERROR;
×
6660
    }
6661

6662
    tBufferDestroy(&local);
6663
    // new col compress
6664
  }
6665

6666
  return 0;
7,796,906✔
6667
}
6668

6669
int32_t tDecompressData(void                *input,       // input
117,825,700✔
6670
                        const SCompressInfo *info,        // compress info
6671
                        void                *output,      // output
6672
                        int32_t              outputSize,  // output size
6673
                        SBuffer             *buffer       // assistant buffer provided by caller, can be NULL
6674
) {
6675
  int32_t code;
6676

6677
  if (!(outputSize >= info->originalSize)) {
117,825,700!
6678
    return TSDB_CODE_INVALID_PARA;
×
6679
  }
6680

6681
  if (info->cmprAlg == NO_COMPRESSION) {
117,825,700✔
6682
    if (!(info->compressedSize == info->originalSize)) {
244!
6683
      return TSDB_CODE_INVALID_PARA;
×
6684
    }
6685
    (void)memcpy(output, input, info->compressedSize);
244✔
6686
  } else if (info->cmprAlg == ONE_STAGE_COMP || info->cmprAlg == TWO_STAGE_COMP) {
164,779,966!
6687
    SBuffer local;
6688

6689
    tBufferInit(&local);
6690
    if (buffer == NULL) {
46,884,467!
6691
      buffer = &local;
×
6692
    }
6693

6694
    if (info->cmprAlg == TWO_STAGE_COMP) {
46,884,467!
6695
      code = tBufferEnsureCapacity(buffer, info->originalSize + COMP_OVERFLOW_BYTES);
46,946,698✔
6696
      if (code) {
46,948,275!
6697
        tBufferDestroy(&local);
6698
        return code;
×
6699
      }
6700
    }
6701

6702
    int32_t decompressedSize = tDataTypes[info->dataType].decompFunc(
46,886,044✔
6703
        input,                                                  // input
6704
        info->compressedSize,                                   // inputSize
46,886,044✔
6705
        info->originalSize / tDataTypes[info->dataType].bytes,  // number of elements
46,886,044✔
6706
        output,                                                 // output
6707
        outputSize,                                             // output size
6708
        info->cmprAlg,                                          // compression algorithm
46,886,044✔
6709
        buffer->data,                                           // helper buffer
6710
        buffer->capacity                                        // extra buffer size
46,886,044✔
6711
    );
6712
    if (decompressedSize < 0) {
46,954,510!
6713
      tBufferDestroy(&local);
6714
      return TSDB_CODE_COMPRESS_ERROR;
×
6715
    }
6716

6717
    if (!(decompressedSize == info->originalSize)) {
46,954,510!
6718
      return TSDB_CODE_COMPRESS_ERROR;
×
6719
    }
6720
    tBufferDestroy(&local);
6721
  } else {
6722
    DEFINE_VAR(info->cmprAlg);
70,940,989✔
6723
    if (l1 == L1_DISABLED && l2 == L2_DISABLED) {
70,940,989✔
6724
      (void)memcpy(output, input, info->compressedSize);
24✔
6725
      return 0;
24✔
6726
    }
6727
    SBuffer local;
6728

6729
    tBufferInit(&local);
6730
    if (buffer == NULL) {
70,940,965!
6731
      buffer = &local;
×
6732
    }
6733
    code = tBufferEnsureCapacity(buffer, info->originalSize + COMP_OVERFLOW_BYTES);
70,940,965✔
6734
    if (code) {
70,940,682!
6735
      return code;
×
6736
    }
6737

6738
    int32_t decompressedSize = tDataCompress[info->dataType].decompFunc(
70,940,682✔
6739
        input,                                                  // input
6740
        info->compressedSize,                                   // inputSize
70,940,682✔
6741
        info->originalSize / tDataTypes[info->dataType].bytes,  // number of elements
70,940,682✔
6742
        output,                                                 // output
6743
        outputSize,                                             // output size
6744
        info->cmprAlg,                                          // compression algorithm
70,940,682✔
6745
        buffer->data,                                           // helper buffer
6746
        buffer->capacity                                        // extra buffer size
70,940,682✔
6747
    );
6748
    if (decompressedSize < 0) {
70,947,682!
6749
      tBufferDestroy(&local);
6750
      return TSDB_CODE_COMPRESS_ERROR;
×
6751
    }
6752

6753
    if (!(decompressedSize == info->originalSize)) {
70,947,682!
6754
      return TSDB_CODE_COMPRESS_ERROR;
×
6755
    }
6756
    tBufferDestroy(&local);
6757
  }
6758

6759
  return 0;
117,902,436✔
6760
}
6761

6762
int32_t tCompressDataToBuffer(void *input, SCompressInfo *info, SBuffer *output, SBuffer *assist) {
7,796,151✔
6763
  int32_t code;
6764

6765
  code = tBufferEnsureCapacity(output, output->size + info->originalSize + COMP_OVERFLOW_BYTES);
7,796,151✔
6766
  if (code) return code;
7,796,159!
6767

6768
  code = tCompressData(input, info, tBufferGetDataEnd(output), output->capacity - output->size, assist);
7,796,159✔
6769
  if (code) return code;
7,796,578!
6770

6771
  output->size += info->compressedSize;
7,796,578✔
6772
  return 0;
7,796,578✔
6773
}
6774

6775
int32_t tDecompressDataToBuffer(void *input, SCompressInfo *info, SBuffer *output, SBuffer *assist) {
46,945,774✔
6776
  int32_t code;
6777

6778
  code = tBufferEnsureCapacity(output, output->size + info->originalSize);
46,945,774✔
6779
  if (code) return code;
46,937,539!
6780

6781
  code = tDecompressData(input, info, tBufferGetDataEnd(output), output->capacity - output->size, assist);
46,937,539✔
6782
  if (code) return code;
46,952,442!
6783

6784
  output->size += info->originalSize;
46,952,442✔
6785
  return 0;
46,952,442✔
6786
}
6787

6788
// handle all types, including var data
6789
void valueSetDatum(SValue *pVal, int8_t type, void *pDatum, uint32_t len) {
2,147,483,647✔
6790
  if (IS_VAR_DATA_TYPE(type) || type == TSDB_DATA_TYPE_DECIMAL) {
2,147,483,647!
6791
    pVal->pData = pDatum;
×
6792
    pVal->nData = len;
×
6793
  } else {
6794
    switch (len) {
2,147,483,647!
6795
      case sizeof(uint8_t):
2,147,483,647✔
6796
        pVal->val = *(uint8_t *)pDatum;
2,147,483,647✔
6797
        break;
2,147,483,647✔
6798
      case sizeof(uint16_t):
1,411,417,264✔
6799
        pVal->val = *(uint16_t *)pDatum;
1,411,417,264✔
6800
        break;
1,411,417,264✔
6801
      case sizeof(uint32_t):
2,147,483,647✔
6802
        pVal->val = *(uint32_t *)pDatum;
2,147,483,647✔
6803
        break;
2,147,483,647✔
6804
      case sizeof(uint64_t):
2,147,483,647✔
6805
        pVal->val = *(uint64_t *)pDatum;
2,147,483,647✔
6806
        break;
2,147,483,647✔
6807
      default:
×
6808
        break;
×
6809
    }
6810
  }
6811
}
2,147,483,647✔
6812

6813
void valueCloneDatum(SValue *pDst, const SValue *pSrc, int8_t type) {
152,237,013✔
6814
  if (IS_VAR_DATA_TYPE(type) || type == TSDB_DATA_TYPE_DECIMAL) {
152,237,013!
6815
    memcpy(pDst->pData, pSrc->pData, pSrc->nData);
203,474✔
6816
    pDst->nData = pSrc->nData;
203,474✔
6817
  } else {
6818
    pDst->val = pSrc->val;
152,033,539✔
6819
  }
6820
}
152,237,013✔
6821
void valueClearDatum(SValue *pVal, int8_t type) {
×
6822
  if (IS_VAR_DATA_TYPE(type) || type == TSDB_DATA_TYPE_DECIMAL) {
×
6823
    taosMemoryFreeClear(pVal->pData);
×
6824
    pVal->nData = 0;
×
6825
  } else {
6826
    pVal->val = 0;
×
6827
  }
6828
}
×
6829

6830
int8_t schemaHasBlob(const STSchema *pSchema) {
15,985,196✔
6831
  if (pSchema == NULL) {
15,985,196!
6832
    return 0;
×
6833
  }
6834
  for (int i = 0; i < pSchema->numOfCols; ++i) {
108,692,931✔
6835
    if (IS_STR_DATA_BLOB(pSchema->columns[i].type)) {
92,707,735!
6836
      return 1;
×
6837
    }
6838
  }
6839
  return 0;
15,985,196✔
6840
}
STATUS · Troubleshooting · Open an Issue · Sales · Support · CAREERS · ENTERPRISE · START FREE · SCHEDULE DEMO
ANNOUNCEMENTS · TWITTER · TOS & SLA · Supported CI Services · What's a CI service? · Automated Testing

© 2026 Coveralls, Inc