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

#define _DEFAULT_SOURCE
S
slguan 已提交
17
#include "tdataformat.h"
S
Shengliang Guan 已提交
18
#include "tcoding.h"
L
Liu Jicong 已提交
19
#include "tdatablock.h"
S
log  
Shengliang Guan 已提交
20
#include "tlog.h"
H
more  
hzcheng 已提交
21

H
Hongze Cheng 已提交
22
typedef struct SKVIdx {
H
Hongze Cheng 已提交
23 24
  int32_t cid;
  int32_t offset;
H
Hongze Cheng 已提交
25
} SKVIdx;
H
Hongze Cheng 已提交
26

H
Hongze Cheng 已提交
27 28 29 30 31 32 33
#pragma pack(push, 1)
typedef struct {
  int16_t nCols;
  SKVIdx  idx[];
} STSKVRow;
#pragma pack(pop)

H
Hongze Cheng 已提交
34
#define TSROW_IS_KV_ROW(r) ((r)->flags & TSROW_KV_ROW)
H
Hongze Cheng 已提交
35 36
#define BIT1_SIZE(n)       (((n)-1) / 8 + 1)
#define BIT2_SIZE(n)       (((n)-1) / 4 + 1)
H
Hongze Cheng 已提交
37 38
#define SET_BIT1(p, i, v)  ((p)[(i) / 8] = (p)[(i) / 8] & (~(((uint8_t)1) << ((i) % 8))) | ((v) << ((i) % 8)))
#define SET_BIT2(p, i, v)  ((p)[(i) / 4] = (p)[(i) / 4] & (~(((uint8_t)3) << ((i) % 4))) | ((v) << ((i) % 4)))
H
Hongze Cheng 已提交
39 40
#define GET_BIT1(p, i)     (((p)[(i) / 8] >> ((i) % 8)) & ((uint8_t)1))
#define GET_BIT2(p, i)     (((p)[(i) / 4] >> ((i) % 4)) & ((uint8_t)3))
H
Hongze Cheng 已提交
41

H
Hongze Cheng 已提交
42 43
static FORCE_INLINE int tSKVIdxCmprFn(const void *p1, const void *p2);

H
Hongze Cheng 已提交
44
// STSRow2
H
Hongze Cheng 已提交
45 46
int32_t tPutTSRow(uint8_t *p, STSRow2 *pRow) {
  int32_t n = 0;
H
Hongze Cheng 已提交
47

H
Hongze Cheng 已提交
48 49 50 51 52
  n += tPutI64(p ? p + n : p, pRow->ts);
  n += tPutI8(p ? p + n : p, pRow->flags);
  n += tPutI32v(p ? p + n : p, pRow->sver);

  ASSERT(pRow->flags & 0xf);
H
Hongze Cheng 已提交
53 54 55 56

  switch (pRow->flags & 0xf) {
    case TSROW_HAS_NONE:
    case TSROW_HAS_NULL:
H
Hongze Cheng 已提交
57
      break;
H
Hongze Cheng 已提交
58
    default:
H
Hongze Cheng 已提交
59
      n += tPutBinary(p ? p + n : p, pRow->pData, pRow->nData);
H
Hongze Cheng 已提交
60
      break;
H
Hongze Cheng 已提交
61
  }
H
Hongze Cheng 已提交
62

H
Hongze Cheng 已提交
63
  return n;
H
Hongze Cheng 已提交
64 65
}

H
Hongze Cheng 已提交
66 67 68
int32_t tGetTSRow(uint8_t *p, STSRow2 *pRow) {
  int32_t n = 0;
  uint8_t flags;
H
Hongze Cheng 已提交
69

H
Hongze Cheng 已提交
70 71 72
  n += tGetI64(p + n, pRow ? &pRow->ts : NULL);
  n += tGetI8(p + n, pRow ? &pRow->flags : &flags);
  n += tGetI32v(p + n, pRow ? &pRow->sver : NULL);
H
Hongze Cheng 已提交
73

H
Hongze Cheng 已提交
74 75
  if (pRow) flags = pRow->flags;
  switch (flags & 0xf) {
H
Hongze Cheng 已提交
76 77
    case TSROW_HAS_NONE:
    case TSROW_HAS_NULL:
H
Hongze Cheng 已提交
78
      break;
H
Hongze Cheng 已提交
79
    default:
H
Hongze Cheng 已提交
80
      n += tGetBinary(p + n, pRow ? &pRow->pData : NULL, pRow ? &pRow->nData : NULL);
H
Hongze Cheng 已提交
81
      break;
H
Hongze Cheng 已提交
82
  }
H
Hongze Cheng 已提交
83

H
Hongze Cheng 已提交
84
  return n;
H
Hongze Cheng 已提交
85 86
}

H
Hongze Cheng 已提交
87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111
int32_t tTSRowDup(const STSRow2 *pRow, STSRow2 **ppRow) {
  (*ppRow) = taosMemoryMalloc(sizeof(*pRow) + pRow->nData);
  if (*ppRow == NULL) {
    terrno = TSDB_CODE_OUT_OF_MEMORY;
    return -1;
  }

  (*ppRow)->ts = pRow->ts;
  (*ppRow)->flags = pRow->flags;
  (*ppRow)->sver = pRow->sver;
  (*ppRow)->nData = pRow->nData;
  if (pRow->nData) {
    (*ppRow)->pData = (uint8_t *)(&(*ppRow)[1]);
    memcpy((*ppRow)->pData, pRow->pData, pRow->nData);
  } else {
    (*ppRow)->pData = NULL;
  }

  return 0;
}

void tTSRowFree(STSRow2 *pRow) {
  if (pRow) taosMemoryFree(pRow);
}

H
Hongze Cheng 已提交
112
int32_t tTSRowGet(const STSRow2 *pRow, STSchema *pTSchema, int32_t iCol, SColVal *pColVal) {
H
Hongze Cheng 已提交
113 114 115 116 117 118 119
  uint32_t       n;
  const uint8_t *p;
  uint8_t        v;
  int32_t        bidx = iCol - 1;
  STColumn      *pTColumn = &pTSchema->columns[iCol];
  STSKVRow      *pTSKVRow;
  SKVIdx        *pKVIdx;
H
Hongze Cheng 已提交
120

H
Hongze Cheng 已提交
121
  ASSERT(iCol != 0);
H
Hongze Cheng 已提交
122 123 124 125 126
  ASSERT(pTColumn->colId != 0);

  ASSERT(pRow->flags & 0xf != 0);
  switch (pRow->flags & 0xf) {
    case TSROW_HAS_NONE:
H
Hongze Cheng 已提交
127
      *pColVal = ColValNONE;
H
Hongze Cheng 已提交
128 129
      return 0;
    case TSROW_HAS_NULL:
H
Hongze Cheng 已提交
130
      *pColVal = ColValNULL;
H
Hongze Cheng 已提交
131 132 133 134 135 136 137
      return 0;
  }

  if (TSROW_IS_KV_ROW(pRow)) {
    ASSERT((pRow->flags & 0xf) != TSROW_HAS_VAL);

    pTSKVRow = (STSKVRow *)pRow->pData;
H
Hongze Cheng 已提交
138 139
    pKVIdx =
        bsearch(&((SKVIdx){.cid = pTColumn->colId}), pTSKVRow->idx, pTSKVRow->nCols, sizeof(SKVIdx), tSKVIdxCmprFn);
H
Hongze Cheng 已提交
140
    if (pKVIdx == NULL) {
H
Hongze Cheng 已提交
141
      *pColVal = ColValNONE;
H
Hongze Cheng 已提交
142
    } else if (pKVIdx->offset < 0) {
H
Hongze Cheng 已提交
143
      *pColVal = ColValNULL;
H
Hongze Cheng 已提交
144 145
    } else {
      p = pRow->pData + sizeof(STSKVRow) + sizeof(SKVIdx) * pTSKVRow->nCols + pKVIdx->offset;
H
Hongze Cheng 已提交
146 147
      pColVal->type = COL_VAL_DATA;
      tGetBinary(p, &pColVal->pData, &pColVal->nData);
H
Hongze Cheng 已提交
148 149 150
    }
  } else {
    // get bitmap
H
Hongze Cheng 已提交
151
    p = pRow->pData;
H
Hongze Cheng 已提交
152
    switch (pRow->flags & 0xf) {
H
Hongze Cheng 已提交
153
      case TSROW_HAS_NULL | TSROW_HAS_NONE:
H
Hongze Cheng 已提交
154 155
        v = GET_BIT1(p, bidx);
        if (v == 0) {
H
Hongze Cheng 已提交
156
          *pColVal = ColValNONE;
H
Hongze Cheng 已提交
157
        } else {
H
Hongze Cheng 已提交
158
          *pColVal = ColValNULL;
H
Hongze Cheng 已提交
159
        }
H
Hongze Cheng 已提交
160
        return 0;
H
Hongze Cheng 已提交
161
      case TSROW_HAS_VAL | TSROW_HAS_NONE:
H
Hongze Cheng 已提交
162 163
        v = GET_BIT1(p, bidx);
        if (v == 1) {
H
Hongze Cheng 已提交
164
          p = p + BIT1_SIZE(pTSchema->numOfCols - 1);
H
Hongze Cheng 已提交
165 166
          break;
        } else {
H
Hongze Cheng 已提交
167
          *pColVal = ColValNONE;
H
Hongze Cheng 已提交
168 169
          return 0;
        }
H
Hongze Cheng 已提交
170
      case TSROW_HAS_VAL | TSROW_HAS_NULL:
H
Hongze Cheng 已提交
171 172
        v = GET_BIT1(p, bidx);
        if (v == 1) {
H
Hongze Cheng 已提交
173
          p = p + BIT1_SIZE(pTSchema->numOfCols - 1);
H
Hongze Cheng 已提交
174
          break;
H
Hongze Cheng 已提交
175
        } else {
H
Hongze Cheng 已提交
176
          *pColVal = ColValNULL;
H
Hongze Cheng 已提交
177
          return 0;
H
Hongze Cheng 已提交
178
        }
H
Hongze Cheng 已提交
179
      case TSROW_HAS_VAL | TSROW_HAS_NULL | TSROW_HAS_NONE:
H
Hongze Cheng 已提交
180 181
        v = GET_BIT2(p, bidx);
        if (v == 0) {
H
Hongze Cheng 已提交
182
          *pColVal = ColValNONE;
H
Hongze Cheng 已提交
183 184
          return 0;
        } else if (v == 1) {
H
Hongze Cheng 已提交
185
          *pColVal = ColValNULL;
H
Hongze Cheng 已提交
186 187
          return 0;
        } else if (v == 2) {
H
Hongze Cheng 已提交
188
          p = p + BIT2_SIZE(pTSchema->numOfCols - 1);
H
Hongze Cheng 已提交
189 190 191 192
          break;
        } else {
          ASSERT(0);
        }
H
Hongze Cheng 已提交
193
      default:
H
Hongze Cheng 已提交
194
        break;
H
Hongze Cheng 已提交
195
    }
H
Hongze Cheng 已提交
196 197 198

    // get real value
    p = p + pTColumn->offset;
H
Hongze Cheng 已提交
199
    pColVal->type = COL_VAL_DATA;
H
Hongze Cheng 已提交
200
    if (IS_VAR_DATA_TYPE(pTColumn->type)) {
H
Hongze Cheng 已提交
201
      tGetBinary(p + pTSchema->flen + *(int32_t *)p, &pColVal->pData, &pColVal->nData);
H
Hongze Cheng 已提交
202
    } else {
H
Hongze Cheng 已提交
203 204
      pColVal->pData = p;
      pColVal->nData = pTColumn->bytes;
H
Hongze Cheng 已提交
205
    }
H
Hongze Cheng 已提交
206
  }
H
Hongze Cheng 已提交
207

H
Hongze Cheng 已提交
208 209 210 211
  return 0;
}

// STSchema
H
Hongze Cheng 已提交
212 213 214 215 216 217 218
int32_t tTSchemaCreate(int32_t sver, SSchema *pSchema, int32_t ncols, STSchema **ppTSchema) {
  *ppTSchema = (STSchema *)taosMemoryMalloc(sizeof(STSchema) + sizeof(STColumn) * ncols);
  if (*ppTSchema == NULL) {
    terrno = TSDB_CODE_OUT_OF_MEMORY;
    return -1;
  }

H
Hongze Cheng 已提交
219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243
  (*ppTSchema)->numOfCols = ncols;
  (*ppTSchema)->version = sver;
  (*ppTSchema)->flen = 0;
  (*ppTSchema)->vlen = 0;
  (*ppTSchema)->tlen = 0;

  for (int32_t iCol = 0; iCol < ncols; iCol++) {
    SSchema  *pColumn = &pSchema[iCol];
    STColumn *pTColumn = &((*ppTSchema)->columns[iCol]);

    pTColumn->colId = pColumn->colId;
    pTColumn->type = pColumn->type;
    pTColumn->flags = pColumn->flags;
    pTColumn->bytes = pColumn->bytes;
    pTColumn->offset = (*ppTSchema)->flen;

    // skip first column
    if (iCol) {
      (*ppTSchema)->flen += TYPE_BYTES[pColumn->type];
      if (IS_VAR_DATA_TYPE(pColumn->type)) {
        (*ppTSchema)->vlen += (pColumn->bytes + 5);
      }
    }
  }

H
Hongze Cheng 已提交
244 245 246
  return 0;
}

H
Hongze Cheng 已提交
247 248 249
void tTSchemaDestroy(STSchema *pTSchema) {
  if (pTSchema) taosMemoryFree(pTSchema);
}
H
Hongze Cheng 已提交
250

H
Hongze Cheng 已提交
251
// STSRowBuilder
H
Hongze Cheng 已提交
252
int32_t tTSRowBuilderInit(STSRowBuilder *pBuilder, int32_t sver, int32_t nCols, SSchema *pSchema) {
H
Hongze Cheng 已提交
253 254
  if (tTSchemaCreate(sver, pSchema, nCols, &pBuilder->pTSchema) < 0) return -1;

H
Hongze Cheng 已提交
255 256
  pBuilder->szBitMap1 = BIT1_SIZE(nCols - 1);
  pBuilder->szBitMap2 = BIT2_SIZE(nCols - 1);
H
Hongze Cheng 已提交
257 258
  pBuilder->szKVBuf =
      sizeof(STSKVRow) + sizeof(SKVIdx) * (nCols - 1) + pBuilder->pTSchema->flen + pBuilder->pTSchema->vlen;
H
Hongze Cheng 已提交
259 260 261
  pBuilder->szTPBuf = pBuilder->szBitMap2 + pBuilder->pTSchema->flen + pBuilder->pTSchema->vlen;
  pBuilder->pKVBuf = taosMemoryMalloc(pBuilder->szKVBuf);
  if (pBuilder->pKVBuf == NULL) {
H
Hongze Cheng 已提交
262 263
    terrno = TSDB_CODE_OUT_OF_MEMORY;
    tTSchemaDestroy(pBuilder->pTSchema);
H
Hongze Cheng 已提交
264
    return -1;
H
Hongze Cheng 已提交
265
  }
H
Hongze Cheng 已提交
266 267 268
  pBuilder->pTPBuf = taosMemoryMalloc(pBuilder->szTPBuf);
  if (pBuilder->pTPBuf == NULL) {
    terrno = TSDB_CODE_OUT_OF_MEMORY;
H
Hongze Cheng 已提交
269 270
    taosMemoryFree(pBuilder->pKVBuf);
    tTSchemaDestroy(pBuilder->pTSchema);
H
Hongze Cheng 已提交
271
    return -1;
H
Hongze Cheng 已提交
272 273
  }

H
Hongze Cheng 已提交
274 275 276
  return 0;
}

H
Hongze Cheng 已提交
277
void tTSRowBuilderClear(STSRowBuilder *pBuilder) {
H
Hongze Cheng 已提交
278 279 280 281
  if (pBuilder->pTPBuf) {
    taosMemoryFree(pBuilder->pTPBuf);
    pBuilder->pTPBuf = NULL;
  }
H
Hongze Cheng 已提交
282 283 284 285 286 287
  if (pBuilder->pKVBuf) {
    taosMemoryFree(pBuilder->pKVBuf);
    pBuilder->pKVBuf = NULL;
  }
  tTSchemaDestroy(pBuilder->pTSchema);
  pBuilder->pTSchema = NULL;
H
Hongze Cheng 已提交
288 289
}

H
Hongze Cheng 已提交
290
void tTSRowBuilderReset(STSRowBuilder *pBuilder) {
H
Hongze Cheng 已提交
291
  for (int32_t iCol = pBuilder->pTSchema->numOfCols - 1; iCol >= 0; iCol--) {
H
Hongze Cheng 已提交
292 293
    STColumn *pTColumn = &pBuilder->pTSchema->columns[iCol];
    COL_CLR_SET(pTColumn->flags);
H
Hongze Cheng 已提交
294 295
  }

H
Hongze Cheng 已提交
296
  pBuilder->iCol = 0;
H
Hongze Cheng 已提交
297
  ((STSKVRow *)pBuilder->pKVBuf)->nCols = 0;
H
Hongze Cheng 已提交
298 299
  pBuilder->vlenKV = 0;
  pBuilder->vlenTP = 0;
H
Hongze Cheng 已提交
300
  pBuilder->row.flags = 0;
H
Hongze Cheng 已提交
301 302
}

H
Hongze Cheng 已提交
303
int32_t tTSRowBuilderPut(STSRowBuilder *pBuilder, int32_t cid, uint8_t *pData, uint32_t nData) {
H
Hongze Cheng 已提交
304 305 306
  STColumn *pTColumn = &pBuilder->pTSchema->columns[pBuilder->iCol];
  uint8_t  *p;
  int32_t   iCol;
H
Hongze Cheng 已提交
307
  STSKVRow *pTSKVRow = (STSKVRow *)pBuilder->pKVBuf;
H
Hongze Cheng 已提交
308

H
Hongze Cheng 已提交
309 310
  // use interp search
  if (pTColumn->colId < cid) {  // right search
H
Hongze Cheng 已提交
311 312
    for (iCol = pBuilder->iCol + 1; iCol < pBuilder->pTSchema->numOfCols; iCol++) {
      pTColumn = &pBuilder->pTSchema->columns[iCol];
H
Hongze Cheng 已提交
313
      if (pTColumn->colId >= cid) break;
H
Hongze Cheng 已提交
314
    }
H
Hongze Cheng 已提交
315
  } else if (pTColumn->colId > cid) {  // left search
H
Hongze Cheng 已提交
316 317
    for (iCol = pBuilder->iCol - 1; iCol >= 0; iCol--) {
      pTColumn = &pBuilder->pTSchema->columns[iCol];
H
Hongze Cheng 已提交
318
      if (pTColumn->colId <= cid) break;
H
Hongze Cheng 已提交
319
    }
H
Hongze Cheng 已提交
320 321
  }

H
Hongze Cheng 已提交
322
  if (pTColumn->colId != cid || COL_IS_SET(pTColumn->flags)) {
H
Hongze Cheng 已提交
323 324 325
    return -1;
  }

H
Hongze Cheng 已提交
326 327
  pBuilder->iCol = iCol;

H
Hongze Cheng 已提交
328 329
  // set value
  if (cid == 0) {
H
Hongze Cheng 已提交
330
    ASSERT(pData && nData == sizeof(TSKEY) && iCol == 0);
H
Hongze Cheng 已提交
331
    pBuilder->row.ts = *(TSKEY *)pData;
H
Hongze Cheng 已提交
332
    pTColumn->flags |= COL_SET_VAL;
H
Hongze Cheng 已提交
333
  } else {
H
Hongze Cheng 已提交
334 335
    if (pData) {
      // set VAL
H
Hongze Cheng 已提交
336

H
Hongze Cheng 已提交
337 338 339 340 341
      pBuilder->row.flags |= TSROW_HAS_VAL;
      pTColumn->flags |= COL_SET_VAL;

      /* KV */
      if (1) {  // avoid KV at some threshold (todo)
H
Hongze Cheng 已提交
342 343
        pTSKVRow->idx[pTSKVRow->nCols].cid = cid;
        pTSKVRow->idx[pTSKVRow->nCols].offset = pBuilder->vlenKV;
H
Hongze Cheng 已提交
344

H
Hongze Cheng 已提交
345 346
        p = pBuilder->pKVBuf + sizeof(STSKVRow) + sizeof(SKVIdx) * (pBuilder->pTSchema->numOfCols - 1) +
            pBuilder->vlenKV;
H
Hongze Cheng 已提交
347 348 349 350 351 352 353 354
        if (IS_VAR_DATA_TYPE(pTColumn->type)) {
          ASSERT(nData <= pTColumn->bytes);
          pBuilder->vlenKV += tPutBinary(p, pData, nData);
        } else {
          ASSERT(nData == pTColumn->bytes);
          memcpy(p, pData, nData);
          pBuilder->vlenKV += nData;
        }
H
Hongze Cheng 已提交
355 356
      }

H
Hongze Cheng 已提交
357 358 359 360 361
      /* TUPLE */
      p = pBuilder->pTPBuf + pBuilder->szBitMap2 + pTColumn->offset;
      if (IS_VAR_DATA_TYPE(pTColumn->type)) {
        ASSERT(nData <= pTColumn->bytes);
        *(int32_t *)p = pBuilder->vlenTP;
H
Hongze Cheng 已提交
362

H
Hongze Cheng 已提交
363
        p = pBuilder->pTPBuf + pBuilder->szBitMap2 + pBuilder->pTSchema->flen + pBuilder->vlenTP;
H
Hongze Cheng 已提交
364
        pBuilder->vlenTP += tPutBinary(p, pData, nData);
H
Hongze Cheng 已提交
365
      } else {
H
Hongze Cheng 已提交
366
        ASSERT(nData == pTColumn->bytes);
H
Hongze Cheng 已提交
367 368
        memcpy(p, pData, nData);
      }
H
Hongze Cheng 已提交
369 370 371
    } else {
      // set NULL

H
Hongze Cheng 已提交
372
      pBuilder->row.flags |= TSROW_HAS_NULL;
H
Hongze Cheng 已提交
373
      pTColumn->flags |= COL_SET_NULL;
H
Hongze Cheng 已提交
374

H
Hongze Cheng 已提交
375 376
      pTSKVRow->idx[pTSKVRow->nCols].cid = cid;
      pTSKVRow->idx[pTSKVRow->nCols].offset = -1;
H
Hongze Cheng 已提交
377
    }
H
Hongze Cheng 已提交
378

H
Hongze Cheng 已提交
379
    pTSKVRow->nCols++;
H
Hongze Cheng 已提交
380 381 382 383 384
  }

  return 0;
}

H
Hongze Cheng 已提交
385 386 387 388 389 390 391 392 393 394
static FORCE_INLINE int tSKVIdxCmprFn(const void *p1, const void *p2) {
  SKVIdx *pKVIdx1 = (SKVIdx *)p1;
  SKVIdx *pKVIdx2 = (SKVIdx *)p2;
  if (pKVIdx1->cid > pKVIdx2->cid) {
    return 1;
  } else if (pKVIdx1->cid < pKVIdx2->cid) {
    return -1;
  }
  return 0;
}
H
Hongze Cheng 已提交
395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413
static void setBitMap(uint8_t *p, STSchema *pTSchema, uint8_t flags) {
  int32_t   bidx;
  STColumn *pTColumn;

  for (int32_t iCol = 1; iCol < pTSchema->numOfCols; iCol++) {
    pTColumn = &pTSchema->columns[iCol];
    bidx = iCol - 1;

    switch (flags) {
      case TSROW_HAS_NULL | TSROW_HAS_NONE:
        if (pTColumn->flags & COL_SET_NULL) {
          SET_BIT1(p, bidx, (uint8_t)1);
        } else {
          SET_BIT1(p, bidx, (uint8_t)0);
        }
        break;
      case TSROW_HAS_VAL | TSROW_HAS_NULL | TSROW_HAS_NONE:
        if (pTColumn->flags & COL_SET_NULL) {
          SET_BIT2(p, bidx, (uint8_t)1);
H
Hongze Cheng 已提交
414
        } else if (pTColumn->flags & COL_SET_VAL) {
H
Hongze Cheng 已提交
415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430
          SET_BIT2(p, bidx, (uint8_t)2);
        } else {
          SET_BIT2(p, bidx, (uint8_t)0);
        }
        break;
      default:
        if (pTColumn->flags & COL_SET_VAL) {
          SET_BIT1(p, bidx, (uint8_t)1);
        } else {
          SET_BIT1(p, bidx, (uint8_t)0);
        }

        break;
    }
  }
}
H
Hongze Cheng 已提交
431
int32_t tTSRowBuilderGetRow(STSRowBuilder *pBuilder, const STSRow2 **ppRow) {
H
Hongze Cheng 已提交
432 433 434
  int32_t   nDataTP, nDataKV;
  uint32_t  flags;
  STSKVRow *pTSKVRow = (STSKVRow *)pBuilder->pKVBuf;
H
Hongze Cheng 已提交
435
  int32_t   nCols = pBuilder->pTSchema->numOfCols;
H
Hongze Cheng 已提交
436 437

  // error not set ts
H
Hongze Cheng 已提交
438
  if (!COL_IS_SET(pBuilder->pTSchema->columns->flags)) {
H
Hongze Cheng 已提交
439 440 441
    return -1;
  }

H
Hongze Cheng 已提交
442 443
  ASSERT(pTSKVRow->nCols < nCols);
  if (pTSKVRow->nCols < nCols - 1) {
H
Hongze Cheng 已提交
444
    pBuilder->row.flags |= TSROW_HAS_NONE;
H
Hongze Cheng 已提交
445
  }
H
Hongze Cheng 已提交
446

H
Hongze Cheng 已提交
447 448 449 450 451
  ASSERT(pBuilder->row.flags & 0xf != 0);
  *(ppRow) = &pBuilder->row;
  switch (pBuilder->row.flags & 0xf) {
    case TSROW_HAS_NONE:
    case TSROW_HAS_NULL:
H
Hongze Cheng 已提交
452 453
      pBuilder->row.nData = 0;
      pBuilder->row.pData = NULL;
H
Hongze Cheng 已提交
454 455
      return 0;
    case TSROW_HAS_NULL | TSROW_HAS_NONE:
H
Hongze Cheng 已提交
456
      nDataTP = pBuilder->szBitMap1;
H
Hongze Cheng 已提交
457
      break;
H
Hongze Cheng 已提交
458
    case TSROW_HAS_VAL:
H
Hongze Cheng 已提交
459
      nDataTP = pBuilder->pTSchema->flen + pBuilder->vlenTP;
H
Hongze Cheng 已提交
460
      break;
H
Hongze Cheng 已提交
461 462
    case TSROW_HAS_VAL | TSROW_HAS_NONE:
    case TSROW_HAS_VAL | TSROW_HAS_NULL:
H
Hongze Cheng 已提交
463
      nDataTP = pBuilder->szBitMap1 + pBuilder->pTSchema->flen + pBuilder->vlenTP;
H
Hongze Cheng 已提交
464
      break;
H
Hongze Cheng 已提交
465
    case TSROW_HAS_VAL | TSROW_HAS_NULL | TSROW_HAS_NONE:
H
Hongze Cheng 已提交
466
      nDataTP = pBuilder->szBitMap2 + pBuilder->pTSchema->flen + pBuilder->vlenTP;
H
Hongze Cheng 已提交
467 468
      break;
    default:
H
Hongze Cheng 已提交
469
      ASSERT(0);
H
Hongze Cheng 已提交
470 471
  }

H
Hongze Cheng 已提交
472 473
  nDataKV = sizeof(STSKVRow) + sizeof(SKVIdx) * pTSKVRow->nCols + pBuilder->vlenKV;
  pBuilder->row.sver = pBuilder->pTSchema->version;
H
Hongze Cheng 已提交
474 475 476
  if (nDataKV < nDataTP) {
    // generate KV row

H
Hongze Cheng 已提交
477 478
    ASSERT(pBuilder->row.flags & 0xf != TSROW_HAS_VAL);

H
Hongze Cheng 已提交
479
    pBuilder->row.flags |= TSROW_KV_ROW;
H
Hongze Cheng 已提交
480
    pBuilder->row.nData = nDataKV;
H
Hongze Cheng 已提交
481
    pBuilder->row.pData = pBuilder->pKVBuf;
H
Hongze Cheng 已提交
482

H
Hongze Cheng 已提交
483
    qsort(pTSKVRow->idx, pTSKVRow->nCols, sizeof(SKVIdx), tSKVIdxCmprFn);
H
Hongze Cheng 已提交
484 485
    if (pTSKVRow->nCols < nCols - 1) {
      memmove(&pTSKVRow->idx[pTSKVRow->nCols], &pTSKVRow->idx[nCols - 1], pBuilder->vlenKV);
H
Hongze Cheng 已提交
486 487
    }
  } else {
H
Hongze Cheng 已提交
488 489 490 491
    // generate TUPLE row

    pBuilder->row.nData = nDataTP;

H
Hongze Cheng 已提交
492 493
    uint8_t *p;
    uint8_t  flags = pBuilder->row.flags & 0xf;
H
Hongze Cheng 已提交
494

H
Hongze Cheng 已提交
495 496
    if (flags == TSROW_HAS_VAL) {
      pBuilder->row.pData = pBuilder->pTPBuf + pBuilder->szBitMap2;
H
Hongze Cheng 已提交
497
    } else {
H
Hongze Cheng 已提交
498 499
      if (flags == TSROW_HAS_VAL | TSROW_HAS_NULL | TSROW_HAS_NONE) {
        pBuilder->row.pData = pBuilder->pTPBuf;
H
Hongze Cheng 已提交
500
      } else {
H
Hongze Cheng 已提交
501
        pBuilder->row.pData = pBuilder->pTPBuf + pBuilder->szBitMap2 - pBuilder->szBitMap1;
H
Hongze Cheng 已提交
502 503
      }

H
Hongze Cheng 已提交
504
      setBitMap(pBuilder->row.pData, pBuilder->pTSchema, flags);
H
Hongze Cheng 已提交
505
    }
H
Hongze Cheng 已提交
506 507 508 509 510
  }

  return 0;
}

H
Hongze Cheng 已提交
511
#if 1  // ====================
512
static void dataColSetNEleNull(SDataCol *pCol, int nEle);
H
Hongze Cheng 已提交
513
int         tdAllocMemForCol(SDataCol *pCol, int maxPoints) {
L
Liu Jicong 已提交
514
  int spaceNeeded = pCol->bytes * maxPoints;
S
Shengliang Guan 已提交
515
  if (IS_VAR_DATA_TYPE(pCol->type)) {
L
Liu Jicong 已提交
516
    spaceNeeded += sizeof(VarDataOffsetT) * maxPoints;
L
Liu Jicong 已提交
517
  }
C
Cary Xu 已提交
518
#ifdef TD_SUPPORT_BITMAP
C
Cary Xu 已提交
519 520 521 522
  int32_t nBitmapBytes = (int32_t)TD_BITMAP_BYTES(maxPoints);
  spaceNeeded += (int)nBitmapBytes;
  // TODO: Currently, the compression of bitmap parts is affiliated to the column data parts, thus allocate 1 more
  // TYPE_BYTES as to comprise complete TYPE_BYTES. Otherwise, invalid read/write would be triggered.
H
Hongze Cheng 已提交
523 524
  // spaceNeeded += TYPE_BYTES[pCol->type]; // the bitmap part is append as a single part since 2022.04.03, thus
  // remove the additional space
C
Cary Xu 已提交
525
#endif
C
Cary Xu 已提交
526

S
Shengliang Guan 已提交
527
  if (pCol->spaceSize < spaceNeeded) {
wafwerar's avatar
wafwerar 已提交
528
    void *ptr = taosMemoryRealloc(pCol->pData, spaceNeeded);
S
Shengliang Guan 已提交
529 530
    if (ptr == NULL) {
      uDebug("malloc failure, size:%" PRId64 " failed, reason:%s", (int64_t)spaceNeeded, strerror(errno));
L
Liu Jicong 已提交
531
      return -1;
L
Liu Jicong 已提交
532 533 534
    } else {
      pCol->pData = ptr;
      pCol->spaceSize = spaceNeeded;
535 536
    }
  }
C
Cary Xu 已提交
537
#ifdef TD_SUPPORT_BITMAP
538

C
Cary Xu 已提交
539 540 541 542
  if (IS_VAR_DATA_TYPE(pCol->type)) {
    pCol->pBitmap = POINTER_SHIFT(pCol->pData, pCol->bytes * maxPoints);
    pCol->dataOff = POINTER_SHIFT(pCol->pBitmap, nBitmapBytes);
  } else {
C
Cary Xu 已提交
543
    pCol->pBitmap = POINTER_SHIFT(pCol->pData, pCol->bytes * maxPoints);
L
Liu Jicong 已提交
544
  }
C
Cary Xu 已提交
545 546 547 548
#else
  if (IS_VAR_DATA_TYPE(pCol->type)) {
    pCol->dataOff = POINTER_SHIFT(pCol->pData, pCol->bytes * maxPoints);
  }
C
Cary Xu 已提交
549
#endif
L
Liu Jicong 已提交
550
  return 0;
551 552
}

H
hzcheng 已提交
553 554 555
/**
 * Duplicate the schema and return a new object
 */
H
Hongze Cheng 已提交
556
STSchema *tdDupSchema(const STSchema *pSchema) {
S
Shengliang Guan 已提交
557
  int       tlen = sizeof(STSchema) + sizeof(STColumn) * schemaNCols(pSchema);
wafwerar's avatar
wafwerar 已提交
558
  STSchema *tSchema = (STSchema *)taosMemoryMalloc(tlen);
H
hzcheng 已提交
559 560
  if (tSchema == NULL) return NULL;

H
Hongze Cheng 已提交
561
  memcpy((void *)tSchema, (void *)pSchema, tlen);
H
hzcheng 已提交
562 563 564 565

  return tSchema;
}

H
TD-27  
hzcheng 已提交
566 567 568
/**
 * Encode a schema to dst, and return the next pointer
 */
H
TD-353  
Hongze Cheng 已提交
569 570 571 572
int tdEncodeSchema(void **buf, STSchema *pSchema) {
  int tlen = 0;
  tlen += taosEncodeFixedI32(buf, schemaVersion(pSchema));
  tlen += taosEncodeFixedI32(buf, schemaNCols(pSchema));
H
TD-166  
hzcheng 已提交
573

H
TD-27  
hzcheng 已提交
574 575
  for (int i = 0; i < schemaNCols(pSchema); i++) {
    STColumn *pCol = schemaColAt(pSchema, i);
H
TD-353  
Hongze Cheng 已提交
576
    tlen += taosEncodeFixedI8(buf, colType(pCol));
C
Cary Xu 已提交
577
    tlen += taosEncodeFixedI8(buf, colFlags(pCol));
H
TD-353  
Hongze Cheng 已提交
578
    tlen += taosEncodeFixedI16(buf, colColId(pCol));
579
    tlen += taosEncodeFixedI16(buf, colBytes(pCol));
H
TD-27  
hzcheng 已提交
580 581
  }

H
TD-353  
Hongze Cheng 已提交
582
  return tlen;
H
TD-27  
hzcheng 已提交
583 584 585 586 587
}

/**
 * Decode a schema from a binary.
 */
H
TD-353  
Hongze Cheng 已提交
588
void *tdDecodeSchema(void *buf, STSchema **pRSchema) {
S
Shengliang Guan 已提交
589 590
  int             version = 0;
  int             numOfCols = 0;
H
TD-353  
Hongze Cheng 已提交
591
  STSchemaBuilder schemaBuilder;
H
TD-27  
hzcheng 已提交
592

H
TD-353  
Hongze Cheng 已提交
593 594
  buf = taosDecodeFixedI32(buf, &version);
  buf = taosDecodeFixedI32(buf, &numOfCols);
H
TD-27  
hzcheng 已提交
595

H
Hongze Cheng 已提交
596 597
  if (tdInitTSchemaBuilder(&schemaBuilder, version) < 0) return NULL;

H
TD-353  
Hongze Cheng 已提交
598
  for (int i = 0; i < numOfCols; i++) {
599
    col_type_t  type = 0;
C
Cary Xu 已提交
600
    int8_t      flags = 0;
601 602
    col_id_t    colId = 0;
    col_bytes_t bytes = 0;
H
TD-353  
Hongze Cheng 已提交
603
    buf = taosDecodeFixedI8(buf, &type);
C
Cary Xu 已提交
604
    buf = taosDecodeFixedI8(buf, &flags);
H
TD-353  
Hongze Cheng 已提交
605
    buf = taosDecodeFixedI16(buf, &colId);
606
    buf = taosDecodeFixedI32(buf, &bytes);
C
Cary Xu 已提交
607
    if (tdAddColToSchema(&schemaBuilder, type, flags, colId, bytes) < 0) {
H
Hongze Cheng 已提交
608 609 610
      tdDestroyTSchemaBuilder(&schemaBuilder);
      return NULL;
    }
H
TD-27  
hzcheng 已提交
611 612
  }

H
TD-353  
Hongze Cheng 已提交
613
  *pRSchema = tdGetSchemaFromBuilder(&schemaBuilder);
H
Hongze Cheng 已提交
614
  tdDestroyTSchemaBuilder(&schemaBuilder);
H
TD-353  
Hongze Cheng 已提交
615
  return buf;
H
Hongze Cheng 已提交
616 617
}

C
Cary Xu 已提交
618
int tdInitTSchemaBuilder(STSchemaBuilder *pBuilder, schema_ver_t version) {
H
Hongze Cheng 已提交
619 620 621
  if (pBuilder == NULL) return -1;

  pBuilder->tCols = 256;
wafwerar's avatar
wafwerar 已提交
622
  pBuilder->columns = (STColumn *)taosMemoryMalloc(sizeof(STColumn) * pBuilder->tCols);
H
Hongze Cheng 已提交
623 624 625 626 627 628 629 630
  if (pBuilder->columns == NULL) return -1;

  tdResetTSchemaBuilder(pBuilder, version);
  return 0;
}

void tdDestroyTSchemaBuilder(STSchemaBuilder *pBuilder) {
  if (pBuilder) {
wafwerar's avatar
wafwerar 已提交
631
    taosMemoryFreeClear(pBuilder->columns);
H
Hongze Cheng 已提交
632 633 634
  }
}

C
Cary Xu 已提交
635
void tdResetTSchemaBuilder(STSchemaBuilder *pBuilder, schema_ver_t version) {
H
Hongze Cheng 已提交
636 637 638
  pBuilder->nCols = 0;
  pBuilder->tlen = 0;
  pBuilder->flen = 0;
T
Tao Liu 已提交
639
  pBuilder->vlen = 0;
H
Hongze Cheng 已提交
640 641 642
  pBuilder->version = version;
}

C
Cary Xu 已提交
643
int32_t tdAddColToSchema(STSchemaBuilder *pBuilder, int8_t type, int8_t flags, col_id_t colId, col_bytes_t bytes) {
644
  if (!isValidDataType(type)) return -1;
H
Hongze Cheng 已提交
645 646 647

  if (pBuilder->nCols >= pBuilder->tCols) {
    pBuilder->tCols *= 2;
wafwerar's avatar
wafwerar 已提交
648
    STColumn *columns = (STColumn *)taosMemoryRealloc(pBuilder->columns, sizeof(STColumn) * pBuilder->tCols);
T
tickduan 已提交
649 650
    if (columns == NULL) return -1;
    pBuilder->columns = columns;
H
Hongze Cheng 已提交
651 652 653 654 655
  }

  STColumn *pCol = &(pBuilder->columns[pBuilder->nCols]);
  colSetType(pCol, type);
  colSetColId(pCol, colId);
C
Cary Xu 已提交
656
  colSetFlags(pCol, flags);
H
Hongze Cheng 已提交
657 658 659
  if (pBuilder->nCols == 0) {
    colSetOffset(pCol, 0);
  } else {
S
Shengliang Guan 已提交
660
    STColumn *pTCol = &(pBuilder->columns[pBuilder->nCols - 1]);
H
Hongze Cheng 已提交
661 662 663 664 665
    colSetOffset(pCol, pTCol->offset + TYPE_BYTES[pTCol->type]);
  }

  if (IS_VAR_DATA_TYPE(type)) {
    colSetBytes(pCol, bytes);
T
Tao Liu 已提交
666 667
    pBuilder->tlen += (TYPE_BYTES[type] + bytes);
    pBuilder->vlen += bytes - sizeof(VarDataLenT);
H
Hongze Cheng 已提交
668 669 670
  } else {
    colSetBytes(pCol, TYPE_BYTES[type]);
    pBuilder->tlen += TYPE_BYTES[type];
T
Tao Liu 已提交
671
    pBuilder->vlen += TYPE_BYTES[type];
H
Hongze Cheng 已提交
672 673 674 675 676 677 678 679 680 681 682 683 684 685 686
  }

  pBuilder->nCols++;
  pBuilder->flen += TYPE_BYTES[type];

  ASSERT(pCol->offset < pBuilder->flen);

  return 0;
}

STSchema *tdGetSchemaFromBuilder(STSchemaBuilder *pBuilder) {
  if (pBuilder->nCols <= 0) return NULL;

  int tlen = sizeof(STSchema) + sizeof(STColumn) * pBuilder->nCols;

wafwerar's avatar
wafwerar 已提交
687
  STSchema *pSchema = (STSchema *)taosMemoryMalloc(tlen);
H
Hongze Cheng 已提交
688 689 690 691 692 693
  if (pSchema == NULL) return NULL;

  schemaVersion(pSchema) = pBuilder->version;
  schemaNCols(pSchema) = pBuilder->nCols;
  schemaTLen(pSchema) = pBuilder->tlen;
  schemaFLen(pSchema) = pBuilder->flen;
T
Tao Liu 已提交
694
  schemaVLen(pSchema) = pBuilder->vlen;
H
Hongze Cheng 已提交
695

C
Cary Xu 已提交
696
#ifdef TD_SUPPORT_BITMAP
C
Cary Xu 已提交
697
  schemaTLen(pSchema) += (int)TD_BITMAP_BYTES(schemaNCols(pSchema));
C
Cary Xu 已提交
698 699
#endif

H
Hongze Cheng 已提交
700 701
  memcpy(schemaColAt(pSchema, 0), pBuilder->columns, sizeof(STColumn) * pBuilder->nCols);

H
TD-27  
hzcheng 已提交
702 703 704
  return pSchema;
}

705
void dataColInit(SDataCol *pDataCol, STColumn *pCol, int maxPoints) {
H
TD-166  
hzcheng 已提交
706 707 708
  pDataCol->type = colType(pCol);
  pDataCol->colId = colColId(pCol);
  pDataCol->bytes = colBytes(pCol);
S
Shengliang Guan 已提交
709
  pDataCol->offset = colOffset(pCol) + 0;  // TD_DATA_ROW_HEAD_SIZE;
H
TD-166  
hzcheng 已提交
710 711 712

  pDataCol->len = 0;
}
C
Cary Xu 已提交
713

L
Liu Jicong 已提交
714 715 716 717 718 719
static FORCE_INLINE const void *tdGetColDataOfRowUnsafe(SDataCol *pCol, int row) {
  if (IS_VAR_DATA_TYPE(pCol->type)) {
    return POINTER_SHIFT(pCol->pData, pCol->dataOff[row]);
  } else {
    return POINTER_SHIFT(pCol->pData, TYPE_BYTES[pCol->type] * row);
  }
H
TD-166  
hzcheng 已提交
720 721
}

H
TD-166  
hzcheng 已提交
722
bool isNEleNull(SDataCol *pCol, int nEle) {
S
Shengliang Guan 已提交
723
  if (isAllRowsNull(pCol)) return true;
724
  for (int i = 0; i < nEle; ++i) {
L
Liu Jicong 已提交
725
    if (!isNull(tdGetColDataOfRowUnsafe(pCol, i), pCol->type)) return false;
H
TD-166  
hzcheng 已提交
726
  }
H
Hongze Cheng 已提交
727
  return true;
H
TD-166  
hzcheng 已提交
728 729
}

C
Cary Xu 已提交
730
void *dataColSetOffset(SDataCol *pCol, int nEle) {
H
TD-166  
hzcheng 已提交
731 732
  ASSERT(((pCol->type == TSDB_DATA_TYPE_BINARY) || (pCol->type == TSDB_DATA_TYPE_NCHAR)));

H
Hongze Cheng 已提交
733
  void *tptr = pCol->pData;
H
TD-166  
hzcheng 已提交
734
  // char *tptr = (char *)(pCol->pData);
H
TD-166  
hzcheng 已提交
735

H
TD-166  
hzcheng 已提交
736
  VarDataOffsetT offset = 0;
737
  for (int i = 0; i < nEle; ++i) {
H
TD-166  
hzcheng 已提交
738
    pCol->dataOff[i] = offset;
H
TD-166  
hzcheng 已提交
739
    offset += varDataTLen(tptr);
H
hzcheng 已提交
740
    tptr = POINTER_SHIFT(tptr, varDataTLen(tptr));
H
TD-166  
hzcheng 已提交
741
  }
C
Cary Xu 已提交
742
  return POINTER_SHIFT(tptr, varDataTLen(tptr));
H
TD-166  
hzcheng 已提交
743 744
}

L
Liu Jicong 已提交
745
SDataCols *tdNewDataCols(int maxCols, int maxRows) {
wafwerar's avatar
wafwerar 已提交
746
  SDataCols *pCols = (SDataCols *)taosMemoryCalloc(1, sizeof(SDataCols));
H
Haojun Liao 已提交
747
  if (pCols == NULL) {
S
Shengliang Guan 已提交
748
    uDebug("malloc failure, size:%" PRId64 " failed, reason:%s", (int64_t)sizeof(SDataCols), strerror(errno));
H
Haojun Liao 已提交
749 750
    return NULL;
  }
H
TD-34  
hzcheng 已提交
751

H
Hongze Cheng 已提交
752
  pCols->maxPoints = maxRows;
L
Liu Jicong 已提交
753 754 755
  pCols->maxCols = maxCols;
  pCols->numOfRows = 0;
  pCols->numOfCols = 0;
C
Cary Xu 已提交
756
  // pCols->bitmapMode = 0; // calloc already set 0
H
Hongze Cheng 已提交
757 758

  if (maxCols > 0) {
wafwerar's avatar
wafwerar 已提交
759
    pCols->cols = (SDataCol *)taosMemoryCalloc(maxCols, sizeof(SDataCol));
H
Hongze Cheng 已提交
760 761 762 763 764 765
    if (pCols->cols == NULL) {
      uDebug("malloc failure, size:%" PRId64 " failed, reason:%s", (int64_t)sizeof(SDataCol) * maxCols,
             strerror(errno));
      tdFreeDataCols(pCols);
      return NULL;
    }
766
#if 0  // no need as calloc used
L
Liu Jicong 已提交
767
    int i;
S
Shengliang Guan 已提交
768
    for (i = 0; i < maxCols; i++) {
L
Liu Jicong 已提交
769
      pCols->cols[i].spaceSize = 0;
L
Liu Jicong 已提交
770
      pCols->cols[i].len = 0;
L
Liu Jicong 已提交
771 772 773
      pCols->cols[i].pData = NULL;
      pCols->cols[i].dataOff = NULL;
    }
774
#endif
H
Hongze Cheng 已提交
775 776
  }

H
TD-34  
hzcheng 已提交
777 778 779
  return pCols;
}

H
Hongze Cheng 已提交
780
int tdInitDataCols(SDataCols *pCols, STSchema *pSchema) {
781 782
  int i;
  int oldMaxCols = pCols->maxCols;
L
Liu Jicong 已提交
783
  if (schemaNCols(pSchema) > oldMaxCols) {
H
Hongze Cheng 已提交
784
    pCols->maxCols = schemaNCols(pSchema);
wafwerar's avatar
wafwerar 已提交
785
    void *ptr = (SDataCol *)taosMemoryRealloc(pCols->cols, sizeof(SDataCol) * pCols->maxCols);
L
Liu Jicong 已提交
786 787
    if (ptr == NULL) return -1;
    pCols->cols = ptr;
788
    for (i = oldMaxCols; i < pCols->maxCols; ++i) {
789 790
      pCols->cols[i].pData = NULL;
      pCols->cols[i].dataOff = NULL;
791
      pCols->cols[i].pBitmap = NULL;
L
Liu Jicong 已提交
792
      pCols->cols[i].spaceSize = 0;
793
    }
L
Liu Jicong 已提交
794
  }
795 796 797
#if 0
  tdResetDataCols(pCols); // redundant loop to reset len/blen to 0, already reset in following dataColInit(...)
#endif
H
Hongze Cheng 已提交
798

799
  pCols->numOfRows = 0;
C
Cary Xu 已提交
800
  pCols->bitmapMode = 0;
H
TD-34  
hzcheng 已提交
801 802
  pCols->numOfCols = schemaNCols(pSchema);

803
  for (i = 0; i < schemaNCols(pSchema); ++i) {
804
    dataColInit(pCols->cols + i, schemaColAt(pSchema, i), pCols->maxPoints);
H
TD-34  
hzcheng 已提交
805
  }
S
Shengliang Guan 已提交
806

H
Hongze Cheng 已提交
807
  return 0;
H
TD-34  
hzcheng 已提交
808 809
}

H
Hongze Cheng 已提交
810
SDataCols *tdFreeDataCols(SDataCols *pCols) {
811
  int i;
H
TD-34  
hzcheng 已提交
812
  if (pCols) {
S
Shengliang Guan 已提交
813
    if (pCols->cols) {
814
      int maxCols = pCols->maxCols;
815
      for (i = 0; i < maxCols; ++i) {
816
        SDataCol *pCol = &pCols->cols[i];
wafwerar's avatar
wafwerar 已提交
817
        taosMemoryFreeClear(pCol->pData);
818
      }
wafwerar's avatar
wafwerar 已提交
819
      taosMemoryFree(pCols->cols);
820 821
      pCols->cols = NULL;
    }
wafwerar's avatar
wafwerar 已提交
822
    taosMemoryFree(pCols);
H
TD-34  
hzcheng 已提交
823
  }
H
Hongze Cheng 已提交
824
  return NULL;
H
TD-34  
hzcheng 已提交
825 826 827
}

void tdResetDataCols(SDataCols *pCols) {
B
Bomin Zhang 已提交
828 829
  if (pCols != NULL) {
    pCols->numOfRows = 0;
C
Cary Xu 已提交
830
    pCols->bitmapMode = 0;
831
    for (int i = 0; i < pCols->maxCols; ++i) {
B
Bomin Zhang 已提交
832 833
      dataColReset(pCols->cols + i);
    }
H
TD-34  
hzcheng 已提交
834 835
  }
}
H
Hongze Cheng 已提交
836

H
Hongze Cheng 已提交
837
SKVRow tdKVRowDup(SKVRow row) {
wafwerar's avatar
wafwerar 已提交
838
  SKVRow trow = taosMemoryMalloc(kvRowLen(row));
H
Hongze Cheng 已提交
839 840
  if (trow == NULL) return NULL;

H
Hongze Cheng 已提交
841
  kvRowCpy(trow, row);
H
Hongze Cheng 已提交
842 843 844
  return trow;
}

S
Shengliang Guan 已提交
845 846 847
static int compareColIdx(const void *a, const void *b) {
  const SColIdx *x = (const SColIdx *)a;
  const SColIdx *y = (const SColIdx *)b;
B
Bomin Zhang 已提交
848 849 850 851 852 853 854 855 856
  if (x->colId > y->colId) {
    return 1;
  }
  if (x->colId < y->colId) {
    return -1;
  }
  return 0;
}

S
Shengliang Guan 已提交
857
void tdSortKVRowByColIdx(SKVRow row) { qsort(kvRowColIdx(row), kvRowNCols(row), sizeof(SColIdx), compareColIdx); }
B
Bomin Zhang 已提交
858

H
TD-90  
Hongze Cheng 已提交
859 860 861 862
int tdSetKVRowDataOfCol(SKVRow *orow, int16_t colId, int8_t type, void *value) {
  SColIdx *pColIdx = NULL;
  SKVRow   row = *orow;
  SKVRow   nrow = NULL;
S
Shengliang Guan 已提交
863
  void    *ptr = taosbsearch(&colId, kvRowColIdx(row), kvRowNCols(row), sizeof(SColIdx), comparTagId, TD_GE);
H
TD-90  
Hongze Cheng 已提交
864

865
  if (ptr == NULL || ((SColIdx *)ptr)->colId > colId) {  // need to add a column value to the row
C
Cary Xu 已提交
866
    int diff = IS_VAR_DATA_TYPE(type) ? varDataTLen(value) : TYPE_BYTES[type];
867 868 869 870
    int nRowLen = kvRowLen(row) + sizeof(SColIdx) + diff;
    int oRowCols = kvRowNCols(row);

    ASSERT(diff > 0);
wafwerar's avatar
wafwerar 已提交
871
    nrow = taosMemoryMalloc(nRowLen);
H
TD-90  
Hongze Cheng 已提交
872 873
    if (nrow == NULL) return -1;

874 875
    kvRowSetLen(nrow, nRowLen);
    kvRowSetNCols(nrow, oRowCols + 1);
H
TD-90  
Hongze Cheng 已提交
876

877 878
    memcpy(kvRowColIdx(nrow), kvRowColIdx(row), sizeof(SColIdx) * oRowCols);
    memcpy(kvRowValues(nrow), kvRowValues(row), kvRowValLen(row));
H
TD-90  
Hongze Cheng 已提交
879

880 881 882
    pColIdx = kvRowColIdxAt(nrow, oRowCols);
    pColIdx->colId = colId;
    pColIdx->offset = kvRowValLen(row);
H
TD-90  
Hongze Cheng 已提交
883

884
    memcpy(kvRowColVal(nrow, pColIdx), value, diff);  // copy new value
H
TD-90  
Hongze Cheng 已提交
885

886
    tdSortKVRowByColIdx(nrow);
H
TD-90  
Hongze Cheng 已提交
887 888

    *orow = nrow;
wafwerar's avatar
wafwerar 已提交
889
    taosMemoryFree(row);
H
TD-90  
Hongze Cheng 已提交
890 891 892 893 894
  } else {
    ASSERT(((SColIdx *)ptr)->colId == colId);
    if (IS_VAR_DATA_TYPE(type)) {
      void *pOldVal = kvRowColVal(row, (SColIdx *)ptr);

S
Shengliang Guan 已提交
895
      if (varDataTLen(value) == varDataTLen(pOldVal)) {  // just update the column value in place
H
TD-90  
Hongze Cheng 已提交
896
        memcpy(pOldVal, value, varDataTLen(value));
897 898
      } else {  // need to reallocate the memory
        int16_t nlen = kvRowLen(row) + (varDataTLen(value) - varDataTLen(pOldVal));
H
TD-90  
Hongze Cheng 已提交
899
        ASSERT(nlen > 0);
wafwerar's avatar
wafwerar 已提交
900
        nrow = taosMemoryMalloc(nlen);
H
TD-90  
Hongze Cheng 已提交
901
        if (nrow == NULL) return -1;
H
TD-90  
Hongze Cheng 已提交
902 903 904 905

        kvRowSetLen(nrow, nlen);
        kvRowSetNCols(nrow, kvRowNCols(row));

906 907 908 909 910 911 912 913
        int zsize = sizeof(SColIdx) * kvRowNCols(row) + ((SColIdx *)ptr)->offset;
        memcpy(kvRowColIdx(nrow), kvRowColIdx(row), zsize);
        memcpy(kvRowColVal(nrow, ((SColIdx *)ptr)), value, varDataTLen(value));
        // Copy left value part
        int lsize = kvRowLen(row) - TD_KV_ROW_HEAD_SIZE - zsize - varDataTLen(pOldVal);
        if (lsize > 0) {
          memcpy(POINTER_SHIFT(nrow, TD_KV_ROW_HEAD_SIZE + zsize + varDataTLen(value)),
                 POINTER_SHIFT(row, TD_KV_ROW_HEAD_SIZE + zsize + varDataTLen(pOldVal)), lsize);
H
TD-90  
Hongze Cheng 已提交
914 915
        }

916 917 918 919 920
        for (int i = 0; i < kvRowNCols(nrow); i++) {
          pColIdx = kvRowColIdxAt(nrow, i);

          if (pColIdx->offset > ((SColIdx *)ptr)->offset) {
            pColIdx->offset = pColIdx->offset - varDataTLen(pOldVal) + varDataTLen(value);
H
TD-90  
Hongze Cheng 已提交
921 922 923 924
          }
        }

        *orow = nrow;
wafwerar's avatar
wafwerar 已提交
925
        taosMemoryFree(row);
H
TD-90  
Hongze Cheng 已提交
926 927 928 929 930 931 932
      }
    } else {
      memcpy(kvRowColVal(row, (SColIdx *)ptr), value, TYPE_BYTES[type]);
    }
  }

  return 0;
H
Hongze Cheng 已提交
933 934
}

H
TD-353  
Hongze Cheng 已提交
935
int tdEncodeKVRow(void **buf, SKVRow row) {
H
Hongze Cheng 已提交
936
  // May change the encode purpose
H
TD-353  
Hongze Cheng 已提交
937 938 939 940 941 942
  if (buf != NULL) {
    kvRowCpy(*buf, row);
    *buf = POINTER_SHIFT(*buf, kvRowLen(row));
  }

  return kvRowLen(row);
H
Hongze Cheng 已提交
943 944
}

H
Hongze Cheng 已提交
945 946
void *tdDecodeKVRow(void *buf, SKVRow *row) {
  *row = tdKVRowDup(buf);
H
TD-353  
Hongze Cheng 已提交
947
  if (*row == NULL) return NULL;
H
Hongze Cheng 已提交
948
  return POINTER_SHIFT(buf, kvRowLen(*row));
H
Hongze Cheng 已提交
949 950
}

H
Hongze Cheng 已提交
951
int tdInitKVRowBuilder(SKVRowBuilder *pBuilder) {
H
Hongze Cheng 已提交
952 953
  pBuilder->tCols = 128;
  pBuilder->nCols = 0;
wafwerar's avatar
wafwerar 已提交
954
  pBuilder->pColIdx = (SColIdx *)taosMemoryMalloc(sizeof(SColIdx) * pBuilder->tCols);
H
Hongze Cheng 已提交
955 956 957
  if (pBuilder->pColIdx == NULL) return -1;
  pBuilder->alloc = 1024;
  pBuilder->size = 0;
wafwerar's avatar
wafwerar 已提交
958
  pBuilder->buf = taosMemoryMalloc(pBuilder->alloc);
H
Hongze Cheng 已提交
959
  if (pBuilder->buf == NULL) {
wafwerar's avatar
wafwerar 已提交
960
    taosMemoryFree(pBuilder->pColIdx);
H
Hongze Cheng 已提交
961 962 963 964 965
    return -1;
  }
  return 0;
}

H
Hongze Cheng 已提交
966
void tdDestroyKVRowBuilder(SKVRowBuilder *pBuilder) {
wafwerar's avatar
wafwerar 已提交
967 968
  taosMemoryFreeClear(pBuilder->pColIdx);
  taosMemoryFreeClear(pBuilder->buf);
H
Hongze Cheng 已提交
969 970
}

H
Hongze Cheng 已提交
971
void tdResetKVRowBuilder(SKVRowBuilder *pBuilder) {
H
Hongze Cheng 已提交
972 973 974 975
  pBuilder->nCols = 0;
  pBuilder->size = 0;
}

H
Hongze Cheng 已提交
976
SKVRow tdGetKVRowFromBuilder(SKVRowBuilder *pBuilder) {
C
Cary Xu 已提交
977
  int tlen = sizeof(SColIdx) * pBuilder->nCols + pBuilder->size;
H
Hongze Cheng 已提交
978 979
  if (tlen == 0) return NULL;

H
Hongze Cheng 已提交
980 981
  tlen += TD_KV_ROW_HEAD_SIZE;

wafwerar's avatar
wafwerar 已提交
982
  SKVRow row = taosMemoryMalloc(tlen);
H
Hongze Cheng 已提交
983 984
  if (row == NULL) return NULL;

H
Hongze Cheng 已提交
985
  kvRowSetNCols(row, pBuilder->nCols);
H
Hongze Cheng 已提交
986
  kvRowSetLen(row, tlen);
H
Hongze Cheng 已提交
987

H
Hongze Cheng 已提交
988 989
  memcpy(kvRowColIdx(row), pBuilder->pColIdx, sizeof(SColIdx) * pBuilder->nCols);
  memcpy(kvRowValues(row), pBuilder->buf, pBuilder->size);
H
Hongze Cheng 已提交
990 991

  return row;
992
}
H
Hongze Cheng 已提交
993
#endif