tdataformat.c 30.4 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

C
Cary Xu 已提交
22 23
static int32_t tGetTagVal(uint8_t *p, STagVal *pTagVal, int8_t isJson);

H
Hongze Cheng 已提交
24 25 26
#pragma pack(push, 1)
typedef struct {
  int16_t nCols;
H
more  
Hongze Cheng 已提交
27
  uint8_t idx[];
H
Hongze Cheng 已提交
28 29 30
} STSKVRow;
#pragma pack(pop)

H
Hongze Cheng 已提交
31 32
#define TSROW_IS_KV_ROW(r) ((r)->flags & TSROW_KV_ROW)

H
Hongze Cheng 已提交
33
// SValue
H
Hongze Cheng 已提交
34
int32_t tPutValue(uint8_t *p, SValue *pValue, int8_t type) {
H
Hongze Cheng 已提交
35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84
  int32_t n = 0;

  if (IS_VAR_DATA_TYPE(type)) {
    n += tPutBinary(p ? p + n : p, pValue->pData, pValue->nData);
  } else {
    switch (type) {
      case TSDB_DATA_TYPE_BOOL:
        n += tPutI8(p ? p + n : p, pValue->i8 ? 1 : 0);
        break;
      case TSDB_DATA_TYPE_TINYINT:
        n += tPutI8(p ? p + n : p, pValue->i8);
        break;
      case TSDB_DATA_TYPE_SMALLINT:
        n += tPutI16(p ? p + n : p, pValue->i16);
        break;
      case TSDB_DATA_TYPE_INT:
        n += tPutI32(p ? p + n : p, pValue->i32);
        break;
      case TSDB_DATA_TYPE_BIGINT:
        n += tPutI64(p ? p + n : p, pValue->i64);
        break;
      case TSDB_DATA_TYPE_FLOAT:
        n += tPutFloat(p ? p + n : p, pValue->f);
        break;
      case TSDB_DATA_TYPE_DOUBLE:
        n += tPutDouble(p ? p + n : p, pValue->d);
        break;
      case TSDB_DATA_TYPE_TIMESTAMP:
        n += tPutI64(p ? p + n : p, pValue->ts);
        break;
      case TSDB_DATA_TYPE_UTINYINT:
        n += tPutU8(p ? p + n : p, pValue->u8);
        break;
      case TSDB_DATA_TYPE_USMALLINT:
        n += tPutU16(p ? p + n : p, pValue->u16);
        break;
      case TSDB_DATA_TYPE_UINT:
        n += tPutU32(p ? p + n : p, pValue->u32);
        break;
      case TSDB_DATA_TYPE_UBIGINT:
        n += tPutU64(p ? p + n : p, pValue->u64);
        break;
      default:
        ASSERT(0);
    }
  }

  return n;
}

H
Hongze Cheng 已提交
85
int32_t tGetValue(uint8_t *p, SValue *pValue, int8_t type) {
H
Hongze Cheng 已提交
86 87 88
  int32_t n = 0;

  if (IS_VAR_DATA_TYPE(type)) {
H
Hongze Cheng 已提交
89
    n += tGetBinary(p, &pValue->pData, pValue ? &pValue->nData : NULL);
H
Hongze Cheng 已提交
90 91 92
  } else {
    switch (type) {
      case TSDB_DATA_TYPE_BOOL:
H
Hongze Cheng 已提交
93
        n += tGetI8(p, &pValue->i8);
H
Hongze Cheng 已提交
94 95
        break;
      case TSDB_DATA_TYPE_TINYINT:
H
Hongze Cheng 已提交
96
        n += tGetI8(p, &pValue->i8);
H
Hongze Cheng 已提交
97 98
        break;
      case TSDB_DATA_TYPE_SMALLINT:
H
Hongze Cheng 已提交
99
        n += tGetI16(p, &pValue->i16);
H
Hongze Cheng 已提交
100 101
        break;
      case TSDB_DATA_TYPE_INT:
H
Hongze Cheng 已提交
102
        n += tGetI32(p, &pValue->i32);
H
Hongze Cheng 已提交
103 104
        break;
      case TSDB_DATA_TYPE_BIGINT:
H
Hongze Cheng 已提交
105
        n += tGetI64(p, &pValue->i64);
H
Hongze Cheng 已提交
106 107
        break;
      case TSDB_DATA_TYPE_FLOAT:
H
Hongze Cheng 已提交
108
        n += tGetFloat(p, &pValue->f);
H
Hongze Cheng 已提交
109 110
        break;
      case TSDB_DATA_TYPE_DOUBLE:
H
Hongze Cheng 已提交
111
        n += tGetDouble(p, &pValue->d);
H
Hongze Cheng 已提交
112 113
        break;
      case TSDB_DATA_TYPE_TIMESTAMP:
H
Hongze Cheng 已提交
114
        n += tGetI64(p, &pValue->ts);
H
Hongze Cheng 已提交
115 116
        break;
      case TSDB_DATA_TYPE_UTINYINT:
H
Hongze Cheng 已提交
117
        n += tGetU8(p, &pValue->u8);
H
Hongze Cheng 已提交
118 119
        break;
      case TSDB_DATA_TYPE_USMALLINT:
H
Hongze Cheng 已提交
120
        n += tGetU16(p, &pValue->u16);
H
Hongze Cheng 已提交
121 122
        break;
      case TSDB_DATA_TYPE_UINT:
H
Hongze Cheng 已提交
123
        n += tGetU32(p, &pValue->u32);
H
Hongze Cheng 已提交
124 125
        break;
      case TSDB_DATA_TYPE_UBIGINT:
H
Hongze Cheng 已提交
126
        n += tGetU64(p, &pValue->u64);
H
Hongze Cheng 已提交
127 128 129 130 131 132 133 134 135
        break;
      default:
        ASSERT(0);
    }
  }

  return n;
}

H
more  
Hongze Cheng 已提交
136 137 138 139 140
int tValueCmprFn(const SValue *pValue1, const SValue *pValue2, int8_t type) {
  // TODO
  return 0;
}

H
Hongze Cheng 已提交
141
// STSRow2 ========================================================================
H
Hongze Cheng 已提交
142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159
static void setBitMap(uint8_t *pb, uint8_t v, int32_t idx, uint8_t flags) {
  if (pb) {
    switch (flags & 0xf) {
      case TSROW_HAS_NULL | TSROW_HAS_NONE:
      case TSROW_HAS_VAL | TSROW_HAS_NONE:
        if (v) {
          SET_BIT1(pb, idx, (uint8_t)1);
        } else {
          SET_BIT1(pb, idx, (uint8_t)0);
        }
        break;
      case TSROW_HAS_VAL | TSROW_HAS_NULL:
        v = v - 1;
        SET_BIT1(pb, idx, v);
        break;
      case TSROW_HAS_VAL | TSROW_HAS_NULL | TSROW_HAS_NONE:
        SET_BIT2(pb, idx, v);
        break;
H
more  
Hongze Cheng 已提交
160

H
Hongze Cheng 已提交
161 162
      default:
        ASSERT(0);
H
more  
Hongze Cheng 已提交
163 164
    }
  }
H
Hongze Cheng 已提交
165 166 167 168 169 170 171 172 173 174 175
}
#define SET_IDX(p, i, n, f)        \
  do {                             \
    if ((f)&TSROW_KV_SMALL) {      \
      ((uint8_t *)(p))[i] = (n);   \
    } else if ((f)&TSROW_KV_MID) { \
      ((uint16_t *)(p))[i] = (n);  \
    } else {                       \
      ((uint32_t *)(p))[i] = (n);  \
    }                              \
  } while (0)
H
more  
Hongze Cheng 已提交
176

H
Hongze Cheng 已提交
177
int32_t tTSRowNew(STSRowBuilder *pBuilder, SArray *pArray, STSchema *pTSchema, STSRow2 **ppRow) {
H
Hongze Cheng 已提交
178 179
  int32_t code = 0;
#if 0
H
Hongze Cheng 已提交
180 181 182 183
  STColumn *pTColumn;
  SColVal  *pColVal;
  int32_t   nColVal = taosArrayGetSize(pArray);
  int32_t   iColVal;
H
more  
Hongze Cheng 已提交
184

H
Hongze Cheng 已提交
185
  ASSERT(nColVal > 0);
H
more  
Hongze Cheng 已提交
186

H
Hongze Cheng 已提交
187 188 189 190 191 192
  // try
  uint8_t  flags = 0;
  uint32_t ntv = 0;
  uint32_t nkv = 0;
  int16_t  nTag = 0;
  uint32_t maxIdx = 0;
H
more  
Hongze Cheng 已提交
193

H
Hongze Cheng 已提交
194 195
  iColVal = 0;
  for (int32_t iColumn = 0; iColumn < pTSchema->numOfCols; iColumn++) {
H
more  
Hongze Cheng 已提交
196 197 198 199 200 201 202
    pTColumn = &pTSchema->columns[iColumn];
    if (iColVal < nColVal) {
      pColVal = (SColVal *)taosArrayGet(pArray, iColVal);
    } else {
      pColVal = NULL;
    }

H
Hongze Cheng 已提交
203 204 205
    if (iColumn == 0) {
      ASSERT(pColVal->cid == pTColumn->colId);
      ASSERT(pTColumn->type == TSDB_DATA_TYPE_TIMESTAMP);
C
Cary Xu 已提交
206
      ASSERT(pTColumn->colId == PRIMARYKEY_TIMESTAMP_COL_ID);
H
Hongze Cheng 已提交
207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232

      iColVal++;
    } else {
      if (pColVal) {
        if (pColVal->cid == pTColumn->colId) {
          iColVal++;

          if (pColVal->isNone) {
            flags |= TSROW_HAS_NONE;
          } else if (pColVal->isNull) {
            flags |= TSROW_HAS_NULL;
            maxIdx = nkv;
            nTag++;
            nkv += tPutI16v(NULL, -pTColumn->colId);
          } else {
            flags |= TSROW_HAS_VAL;
            maxIdx = nkv;
            nTag++;
            nkv += tPutI16v(NULL, pTColumn->colId);
            nkv += tPutValue(NULL, &pColVal->value, pTColumn->type);
            if (IS_VAR_DATA_TYPE(pTColumn->type)) {
              ntv += tPutValue(NULL, &pColVal->value, pTColumn->type);
            }
          }
        } else if (pColVal->cid > pTColumn->colId) {
          flags |= TSROW_HAS_NONE;
H
more  
Hongze Cheng 已提交
233
        } else {
H
Hongze Cheng 已提交
234
          ASSERT(0);
H
more  
Hongze Cheng 已提交
235 236
        }
      } else {
H
Hongze Cheng 已提交
237
        flags |= TSROW_HAS_NONE;
H
more  
Hongze Cheng 已提交
238 239
      }
    }
H
Hongze Cheng 已提交
240
  }
H
more  
Hongze Cheng 已提交
241

C
Cary Xu 已提交
242
  ASSERT(flags);
H
more  
Hongze Cheng 已提交
243

H
Hongze Cheng 已提交
244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265
  // decide
  uint32_t nData = 0;
  uint32_t nDataT = 0;
  uint32_t nDataK = 0;
  if (flags == TSROW_HAS_NONE || flags == TSROW_HAS_NULL) {
    nData = 0;
  } else {
    switch (flags) {
      case TSROW_HAS_VAL:
        nDataT = pTSchema->flen + ntv;
        break;
      case TSROW_HAS_NULL | TSROW_HAS_NONE:
        nDataT = BIT1_SIZE(pTSchema->numOfCols - 1);
        break;
      case TSROW_HAS_VAL | TSROW_HAS_NONE:
      case TSROW_HAS_VAL | TSROW_HAS_NULL:
        nDataT = BIT1_SIZE(pTSchema->numOfCols - 1) + pTSchema->flen + ntv;
        break;
      case TSROW_HAS_VAL | TSROW_HAS_NULL | TSROW_HAS_NONE:
        nDataT = BIT2_SIZE(pTSchema->numOfCols - 1) + pTSchema->flen + ntv;
        break;
      default:
C
Cary Xu 已提交
266 267
        break;
        ASSERT(0);
H
more  
Hongze Cheng 已提交
268 269
    }

H
Hongze Cheng 已提交
270 271 272 273 274 275 276 277 278 279
    uint8_t tflags = 0;
    if (maxIdx <= UINT8_MAX) {
      nDataK = sizeof(STSKVRow) + sizeof(uint8_t) * nTag + nkv;
      tflags |= TSROW_KV_SMALL;
    } else if (maxIdx <= UINT16_MAX) {
      nDataK = sizeof(STSKVRow) + sizeof(uint16_t) * nTag + nkv;
      tflags |= TSROW_KV_MID;
    } else {
      nDataK = sizeof(STSKVRow) + sizeof(uint32_t) * nTag + nkv;
      tflags |= TSROW_KV_BIG;
H
more  
Hongze Cheng 已提交
280 281
    }

C
Cary Xu 已提交
282
    if (nDataT <= nDataK) {
H
Hongze Cheng 已提交
283
      nData = nDataT;
H
more  
Hongze Cheng 已提交
284
    } else {
H
Hongze Cheng 已提交
285 286
      nData = nDataK;
      flags |= tflags;
H
more  
Hongze Cheng 已提交
287 288
    }
  }
H
more  
Hongze Cheng 已提交
289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306

  // alloc
  if (pBuilder) {
    // create from a builder
    if (nData == 0) {
      pBuilder->tsRow.nData = 0;
      pBuilder->tsRow.pData = NULL;
    } else {
      if (pBuilder->szBuf < nData) {
        uint8_t *p = taosMemoryRealloc(pBuilder->pBuf, nData);
        if (p == NULL) {
          code = TSDB_CODE_OUT_OF_MEMORY;
          goto _exit;
        }
        pBuilder->pBuf = p;
        pBuilder->szBuf = nData;
      }

H
Hongze Cheng 已提交
307
      pBuilder->tsRow.nData = nData;
H
more  
Hongze Cheng 已提交
308 309 310 311 312 313 314 315
      pBuilder->tsRow.pData = pBuilder->pBuf;
    }

    *ppRow = &pBuilder->tsRow;
  } else {
    // create a new one
    *ppRow = (STSRow2 *)taosMemoryMalloc(sizeof(STSRow2));
    if (*ppRow == NULL) {
H
more  
Hongze Cheng 已提交
316 317 318
      code = TSDB_CODE_OUT_OF_MEMORY;
      goto _exit;
    }
H
Hongze Cheng 已提交
319 320 321 322 323
    if (nData == 0) {
      (*ppRow)->nData = 0;
      (*ppRow)->pData = NULL;
    } else {
      (*ppRow)->nData = nData;
H
more  
Hongze Cheng 已提交
324 325
      (*ppRow)->pData = taosMemoryMalloc(nData);
      if ((*ppRow)->pData == NULL) {
H
Hongze Cheng 已提交
326
        taosMemoryFree(*ppRow);
H
more  
Hongze Cheng 已提交
327 328 329 330
        code = TSDB_CODE_OUT_OF_MEMORY;
        goto _exit;
      }
    }
H
more  
Hongze Cheng 已提交
331
  }
H
more  
Hongze Cheng 已提交
332

H
Hongze Cheng 已提交
333 334 335
  // build
  (*ppRow)->flags = flags;
  (*ppRow)->sver = pTSchema->version;
H
more  
Hongze Cheng 已提交
336

H
Hongze Cheng 已提交
337 338 339 340 341 342 343 344 345 346 347 348 349 350
  pColVal = (SColVal *)taosArrayGet(pArray, 0);
  (*ppRow)->ts = pColVal->value.ts;

  if ((*ppRow)->pData) {
    STSKVRow *pTSKVRow = NULL;
    uint8_t  *pidx = NULL;
    uint8_t  *pkv = NULL;
    uint8_t  *pb = NULL;
    uint8_t  *pf = NULL;
    uint8_t  *ptv = NULL;
    nkv = 0;
    ntv = 0;
    iColVal = 1;

C
Cary Xu 已提交
351
    if ((flags & 0xf0) == 0) {
H
Hongze Cheng 已提交
352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371
      switch (flags & 0xf) {
        case TSROW_HAS_VAL:
          pf = (*ppRow)->pData;
          ptv = pf + pTSchema->flen;
          break;
        case TSROW_HAS_NULL | TSROW_HAS_NONE:
          pb = (*ppRow)->pData;
          break;
        case TSROW_HAS_VAL | TSROW_HAS_NONE:
        case TSROW_HAS_VAL | TSROW_HAS_NULL:
          pb = (*ppRow)->pData;
          pf = pb + BIT1_SIZE(pTSchema->numOfCols - 1);
          ptv = pf + pTSchema->flen;
          break;
        case TSROW_HAS_VAL | TSROW_HAS_NULL | TSROW_HAS_NONE:
          pb = (*ppRow)->pData;
          pf = pb + BIT2_SIZE(pTSchema->numOfCols - 1);
          ptv = pf + pTSchema->flen;
          break;
        default:
C
Cary Xu 已提交
372
          ASSERT(0);
C
Cary Xu 已提交
373
          break;
H
Hongze Cheng 已提交
374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416
      }
    } else {
      pTSKVRow = (STSKVRow *)(*ppRow)->pData;
      pTSKVRow->nCols = 0;
      pidx = pTSKVRow->idx;
      if (flags & TSROW_KV_SMALL) {
        pkv = pidx + sizeof(uint8_t) * nTag;
      } else if (flags & TSROW_KV_MID) {
        pkv = pidx + sizeof(uint16_t) * nTag;
      } else {
        pkv = pidx + sizeof(uint32_t) * nTag;
      }
    }

    for (int32_t iColumn = 1; iColumn < pTSchema->numOfCols; iColumn++) {
      pTColumn = &pTSchema->columns[iColumn];
      if (iColVal < nColVal) {
        pColVal = (SColVal *)taosArrayGet(pArray, iColVal);
      } else {
        pColVal = NULL;
      }

      if (pColVal) {
        if (pColVal->cid == pTColumn->colId) {
          iColVal++;

          if (pColVal->isNone) {
            goto _set_none;
          } else if (pColVal->isNull) {
            goto _set_null;
          } else {
            goto _set_value;
          }
        } else if (pColVal->cid > pTColumn->colId) {
          goto _set_none;
        } else {
          ASSERT(0);
        }
      } else {
        goto _set_none;
      }

    _set_none:
C
Cary Xu 已提交
417
      if ((flags & 0xf0) == 0) {
H
Hongze Cheng 已提交
418
        setBitMap(pb, 0, iColumn - 1, flags);
H
Hongze Cheng 已提交
419
        if (flags & TSROW_HAS_VAL) {  // set 0
C
Cary Xu 已提交
420 421 422 423 424 425
          if (IS_VAR_DATA_TYPE(pTColumn->type)) {
            *(VarDataOffsetT *)(pf + pTColumn->offset) = 0;
          } else {
            tPutValue(pf + pTColumn->offset, &((SValue){0}), pTColumn->type);
          }
        }
H
Hongze Cheng 已提交
426 427 428 429
      }
      continue;

    _set_null:
C
Cary Xu 已提交
430
      if ((flags & 0xf0) == 0) {
H
Hongze Cheng 已提交
431
        setBitMap(pb, 1, iColumn - 1, flags);
H
Hongze Cheng 已提交
432
        if (flags & TSROW_HAS_VAL) {  // set 0
C
Cary Xu 已提交
433 434 435 436 437 438
          if (IS_VAR_DATA_TYPE(pTColumn->type)) {
            *(VarDataOffsetT *)(pf + pTColumn->offset) = 0;
          } else {
            tPutValue(pf + pTColumn->offset, &((SValue){0}), pTColumn->type);
          }
        }
H
Hongze Cheng 已提交
439 440 441 442 443 444
      } else {
        SET_IDX(pidx, pTSKVRow->nCols, nkv, flags);
        pTSKVRow->nCols++;
        nkv += tPutI16v(pkv + nkv, -pTColumn->colId);
      }
      continue;
H
more  
Hongze Cheng 已提交
445

H
Hongze Cheng 已提交
446
    _set_value:
C
Cary Xu 已提交
447
      if ((flags & 0xf0) == 0) {
H
Hongze Cheng 已提交
448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463
        setBitMap(pb, 2, iColumn - 1, flags);

        if (IS_VAR_DATA_TYPE(pTColumn->type)) {
          *(VarDataOffsetT *)(pf + pTColumn->offset) = ntv;
          ntv += tPutValue(ptv + ntv, &pColVal->value, pTColumn->type);
        } else {
          tPutValue(pf + pTColumn->offset, &pColVal->value, pTColumn->type);
        }
      } else {
        SET_IDX(pidx, pTSKVRow->nCols, nkv, flags);
        pTSKVRow->nCols++;
        nkv += tPutI16v(pkv + nkv, pColVal->cid);
        nkv += tPutValue(pkv + nkv, &pColVal->value, pTColumn->type);
      }
      continue;
    }
H
more  
Hongze Cheng 已提交
464 465
  }

H
Hongze Cheng 已提交
466
#endif
H
more  
Hongze Cheng 已提交
467
_exit:
H
Hongze Cheng 已提交
468 469 470
  return code;
}

H
Hongze Cheng 已提交
471 472
int32_t tTSRowClone(const STSRow2 *pRow, STSRow2 **ppRow) {
  int32_t code = 0;
H
Hongze Cheng 已提交
473
  int32_t rLen;
H
Hongze Cheng 已提交
474

H
Hongze Cheng 已提交
475 476
  TSROW_LEN(pRow, rLen);
  (*ppRow) = (STSRow2 *)taosMemoryMalloc(rLen);
H
Hongze Cheng 已提交
477
  if (*ppRow == NULL) {
H
Hongze Cheng 已提交
478 479
    code = TSDB_CODE_OUT_OF_MEMORY;
    goto _exit;
H
Hongze Cheng 已提交
480
  }
H
Hongze Cheng 已提交
481
  memcpy(*ppRow, pRow, rLen);
H
Hongze Cheng 已提交
482

H
Hongze Cheng 已提交
483 484
_exit:
  return code;
H
Hongze Cheng 已提交
485 486 487
}

void tTSRowFree(STSRow2 *pRow) {
H
Hongze Cheng 已提交
488 489 490
  if (pRow) {
    taosMemoryFree(pRow);
  }
H
Hongze Cheng 已提交
491 492
}

H
Hongze Cheng 已提交
493
void tTSRowGet(STSRow2 *pRow, STSchema *pTSchema, int32_t iCol, SColVal *pColVal) {
H
Hongze Cheng 已提交
494
#if 0
C
Cary Xu 已提交
495
  uint8_t   isTuple = ((pRow->flags & 0xf0) == 0) ? 1 : 0;
H
Hongze Cheng 已提交
496
  STColumn *pTColumn = &pTSchema->columns[iCol];
H
Hongze Cheng 已提交
497 498
  uint8_t   flags = pRow->flags & (uint8_t)0xf;
  SValue    value;
H
Hongze Cheng 已提交
499

H
Hongze Cheng 已提交
500
  ASSERT(iCol < pTSchema->numOfCols);
C
Cary Xu 已提交
501
  ASSERT(flags);
H
Hongze Cheng 已提交
502
  ASSERT(pRow->sver == pTSchema->version);
H
Hongze Cheng 已提交
503

H
Hongze Cheng 已提交
504 505 506
  if (iCol == 0) {
    value.ts = pRow->ts;
    goto _return_value;
H
Hongze Cheng 已提交
507 508
  }

H
Hongze Cheng 已提交
509
  if (flags == TSROW_HAS_NONE) {
H
more  
Hongze Cheng 已提交
510
    goto _return_none;
C
Cary Xu 已提交
511
  } else if (flags == TSROW_HAS_NULL) {
H
more  
Hongze Cheng 已提交
512
    goto _return_null;
H
Hongze Cheng 已提交
513
  }
H
Hongze Cheng 已提交
514

H
Hongze Cheng 已提交
515 516 517 518 519 520 521 522 523 524 525 526 527 528
  ASSERT(pRow->nData && pRow->pData);

  if (isTuple) {
    uint8_t *pb = pRow->pData;
    uint8_t *pf = NULL;
    uint8_t *pv = NULL;
    uint8_t *p;
    uint8_t  b;

    // bit
    switch (flags) {
      case TSROW_HAS_VAL:
        pf = pb;
        break;
H
Hongze Cheng 已提交
529
      case TSROW_HAS_NULL | TSROW_HAS_NONE:
H
Hongze Cheng 已提交
530 531 532
        b = GET_BIT1(pb, iCol - 1);
        if (b == 0) {
          goto _return_none;
H
Hongze Cheng 已提交
533
        } else {
H
Hongze Cheng 已提交
534
          goto _return_null;
H
Hongze Cheng 已提交
535
        }
H
Hongze Cheng 已提交
536
      case TSROW_HAS_VAL | TSROW_HAS_NONE:
H
Hongze Cheng 已提交
537 538 539
        b = GET_BIT1(pb, iCol - 1);
        if (b == 0) {
          goto _return_none;
H
Hongze Cheng 已提交
540
        } else {
H
Hongze Cheng 已提交
541 542
          pf = pb + BIT1_SIZE(pTSchema->numOfCols - 1);
          break;
H
Hongze Cheng 已提交
543
        }
H
Hongze Cheng 已提交
544
      case TSROW_HAS_VAL | TSROW_HAS_NULL:
H
Hongze Cheng 已提交
545 546 547
        b = GET_BIT1(pb, iCol - 1);
        if (b == 0) {
          goto _return_null;
H
Hongze Cheng 已提交
548
        } else {
H
Hongze Cheng 已提交
549 550
          pf = pb + BIT1_SIZE(pTSchema->numOfCols - 1);
          break;
H
Hongze Cheng 已提交
551
        }
H
Hongze Cheng 已提交
552
      case TSROW_HAS_VAL | TSROW_HAS_NULL | TSROW_HAS_NONE:
H
Hongze Cheng 已提交
553 554 555 556 557
        b = GET_BIT2(pb, iCol - 1);
        if (b == 0) {
          goto _return_none;
        } else if (b == 1) {
          goto _return_null;
H
Hongze Cheng 已提交
558
        } else {
H
Hongze Cheng 已提交
559 560
          pf = pb + BIT2_SIZE(pTSchema->numOfCols - 1);
          break;
H
Hongze Cheng 已提交
561
        }
H
Hongze Cheng 已提交
562
      default:
H
Hongze Cheng 已提交
563
        ASSERT(0);
H
Hongze Cheng 已提交
564
    }
H
Hongze Cheng 已提交
565

H
Hongze Cheng 已提交
566 567 568
    ASSERT(pf);

    p = pf + pTColumn->offset;
H
Hongze Cheng 已提交
569
    if (IS_VAR_DATA_TYPE(pTColumn->type)) {
H
Hongze Cheng 已提交
570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591
      pv = pf + pTSchema->flen;
      p = pv + *(VarDataOffsetT *)p;
    }
    tGetValue(p, &value, pTColumn->type);
    goto _return_value;
  } else {
    STSKVRow *pRowK = (STSKVRow *)pRow->pData;
    int16_t   lidx = 0;
    int16_t   ridx = pRowK->nCols - 1;
    uint8_t  *p;
    int16_t   midx;
    uint32_t  n;
    int16_t   cid;

    ASSERT(pRowK->nCols > 0);

    if (pRow->flags & TSROW_KV_SMALL) {
      p = pRow->pData + sizeof(STSKVRow) + sizeof(uint8_t) * pRowK->nCols;
    } else if (pRow->flags & TSROW_KV_MID) {
      p = pRow->pData + sizeof(STSKVRow) + sizeof(uint16_t) * pRowK->nCols;
    } else if (pRow->flags & TSROW_KV_BIG) {
      p = pRow->pData + sizeof(STSKVRow) + sizeof(uint32_t) * pRowK->nCols;
H
Hongze Cheng 已提交
592
    } else {
H
Hongze Cheng 已提交
593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621
      ASSERT(0);
    }
    while (lidx <= ridx) {
      midx = (lidx + ridx) / 2;

      if (pRow->flags & TSROW_KV_SMALL) {
        n = ((uint8_t *)pRowK->idx)[midx];
      } else if (pRow->flags & TSROW_KV_MID) {
        n = ((uint16_t *)pRowK->idx)[midx];
      } else {
        n = ((uint32_t *)pRowK->idx)[midx];
      }

      n += tGetI16v(p + n, &cid);

      if (TABS(cid) == pTColumn->colId) {
        if (cid < 0) {
          goto _return_null;
        } else {
          n += tGetValue(p + n, &value, pTColumn->type);
          goto _return_value;
        }

        return;
      } else if (TABS(cid) > pTColumn->colId) {
        ridx = midx - 1;
      } else {
        lidx = midx + 1;
      }
H
Hongze Cheng 已提交
622
    }
H
Hongze Cheng 已提交
623 624 625

    // not found, return NONE
    goto _return_none;
H
Hongze Cheng 已提交
626
  }
H
Hongze Cheng 已提交
627

H
Hongze Cheng 已提交
628
_return_none:
H
Hongze Cheng 已提交
629
  *pColVal = COL_VAL_NONE(pTColumn->colId, pTColumn->type);
H
Hongze Cheng 已提交
630 631 632
  return;

_return_null:
H
Hongze Cheng 已提交
633
  *pColVal = COL_VAL_NULL(pTColumn->colId, pTColumn->type);
H
Hongze Cheng 已提交
634 635 636
  return;

_return_value:
H
Hongze Cheng 已提交
637
  *pColVal = COL_VAL_VALUE(pTColumn->colId, pTColumn->type, value);
H
Hongze Cheng 已提交
638
  return;
H
Hongze Cheng 已提交
639
#endif
H
Hongze Cheng 已提交
640 641 642 643
}

int32_t tTSRowToArray(STSRow2 *pRow, STSchema *pTSchema, SArray **ppArray) {
  int32_t code = 0;
H
Hongze Cheng 已提交
644
#if 0
H
Hongze Cheng 已提交
645 646 647 648 649 650 651 652 653 654 655 656 657
  SColVal cv;

  (*ppArray) = taosArrayInit(pTSchema->numOfCols, sizeof(SColVal));
  if (*ppArray == NULL) {
    code = TSDB_CODE_OUT_OF_MEMORY;
    goto _exit;
  }

  for (int32_t iColumn = 0; iColumn < pTSchema->numOfCols; iColumn++) {
    tTSRowGet(pRow, pTSchema, iColumn, &cv);
    taosArrayPush(*ppArray, &cv);
  }

H
Hongze Cheng 已提交
658
#endif
H
Hongze Cheng 已提交
659 660 661 662 663
_exit:
  return code;
}

int32_t tPutTSRow(uint8_t *p, STSRow2 *pRow) {
H
Hongze Cheng 已提交
664
  int32_t n;
H
Hongze Cheng 已提交
665

H
Hongze Cheng 已提交
666 667 668
  TSROW_LEN(pRow, n);
  if (p) {
    memcpy(p, pRow, n);
H
Hongze Cheng 已提交
669 670 671 672 673
  }

  return n;
}

H
Hongze Cheng 已提交
674 675
int32_t tGetTSRow(uint8_t *p, STSRow2 **ppRow) {
  int32_t n;
H
Hongze Cheng 已提交
676

H
Hongze Cheng 已提交
677 678
  *ppRow = (STSRow2 *)p;
  TSROW_LEN(*ppRow, n);
H
Hongze Cheng 已提交
679 680

  return n;
H
Hongze Cheng 已提交
681 682 683
}

// STSchema
H
Hongze Cheng 已提交
684 685 686 687 688 689 690
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 已提交
691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706 707 708 709 710 711 712 713 714 715
  (*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 已提交
716 717 718
  return 0;
}

H
Hongze Cheng 已提交
719 720 721
void tTSchemaDestroy(STSchema *pTSchema) {
  if (pTSchema) taosMemoryFree(pTSchema);
}
H
Hongze Cheng 已提交
722

H
Hongze Cheng 已提交
723
// STSRowBuilder
H
Hongze Cheng 已提交
724

H
more  
Hongze Cheng 已提交
725
// STag
H
Hongze Cheng 已提交
726 727
static int tTagValCmprFn(const void *p1, const void *p2) {
  if (((STagVal *)p1)->cid < ((STagVal *)p2)->cid) {
H
Hongze Cheng 已提交
728
    return -1;
H
Hongze Cheng 已提交
729
  } else if (((STagVal *)p1)->cid > ((STagVal *)p2)->cid) {
H
Hongze Cheng 已提交
730 731
    return 1;
  }
H
Hongze Cheng 已提交
732

H
Hongze Cheng 已提交
733 734
  return 0;
}
H
Hongze Cheng 已提交
735 736 737
static int tTagValJsonCmprFn(const void *p1, const void *p2) {
  return strcmp(((STagVal *)p1)[0].pKey, ((STagVal *)p2)[0].pKey);
}
C
Cary Xu 已提交
738 739 740 741 742 743 744

static void debugPrintTagVal(int8_t type, const void *val, int32_t vlen, const char *tag, int32_t ln) {
  switch (type) {
    case TSDB_DATA_TYPE_JSON:
    case TSDB_DATA_TYPE_VARCHAR:
    case TSDB_DATA_TYPE_NCHAR: {
      char tmpVal[32] = {0};
745
      strncpy(tmpVal, val, vlen > 31 ? 31 : vlen);
C
Cary Xu 已提交
746 747 748 749 750 751 752 753 754 755 756 757 758 759 760 761 762 763 764 765 766 767 768 769 770 771 772 773 774 775 776 777 778 779 780 781 782 783
      printf("%s:%d type:%d vlen:%d, val:\"%s\"\n", tag, ln, (int32_t)type, vlen, tmpVal);
    } break;
    case TSDB_DATA_TYPE_FLOAT:
      printf("%s:%d type:%d vlen:%d, val:%f\n", tag, ln, (int32_t)type, vlen, *(float *)val);
      break;
    case TSDB_DATA_TYPE_DOUBLE:
      printf("%s:%d type:%d vlen:%d, val:%lf\n", tag, ln, (int32_t)type, vlen, *(double *)val);
      break;
    case TSDB_DATA_TYPE_BOOL:
      printf("%s:%d type:%d vlen:%d, val:%" PRIu8 "\n", tag, ln, (int32_t)type, vlen, *(uint8_t *)val);
      break;
    case TSDB_DATA_TYPE_TINYINT:
      printf("%s:%d type:%d vlen:%d, val:%" PRIi8 "\n", tag, ln, (int32_t)type, vlen, *(int8_t *)val);
      break;
    case TSDB_DATA_TYPE_SMALLINT:
      printf("%s:%d type:%d vlen:%d, val:%" PRIi16 "\n", tag, ln, (int32_t)type, vlen, *(int16_t *)val);
      break;
    case TSDB_DATA_TYPE_INT:
      printf("%s:%d type:%d vlen:%d, val:%" PRIi32 "\n", tag, ln, (int32_t)type, vlen, *(int32_t *)val);
      break;
    case TSDB_DATA_TYPE_BIGINT:
      printf("%s:%d type:%d vlen:%d, val:%" PRIi64 "\n", tag, ln, (int32_t)type, vlen, *(int64_t *)val);
      break;
    case TSDB_DATA_TYPE_TIMESTAMP:
      printf("%s:%d type:%d vlen:%d, val:%" PRIi64 "\n", tag, ln, (int32_t)type, vlen, *(int64_t *)val);
      break;
    case TSDB_DATA_TYPE_UTINYINT:
      printf("%s:%d type:%d vlen:%d, val:%" PRIu8 "\n", tag, ln, (int32_t)type, vlen, *(uint8_t *)val);
      break;
    case TSDB_DATA_TYPE_USMALLINT:
      printf("%s:%d type:%d vlen:%d, val:%" PRIu16 "\n", tag, ln, (int32_t)type, vlen, *(uint16_t *)val);
      break;
    case TSDB_DATA_TYPE_UINT:
      printf("%s:%d type:%d vlen:%d, val:%" PRIu32 "\n", tag, ln, (int32_t)type, vlen, *(uint32_t *)val);
      break;
    case TSDB_DATA_TYPE_UBIGINT:
      printf("%s:%d type:%d vlen:%d, val:%" PRIu64 "\n", tag, ln, (int32_t)type, vlen, *(uint64_t *)val);
      break;
wmmhello's avatar
wmmhello 已提交
784 785 786
    case TSDB_DATA_TYPE_NULL:
      printf("%s:%d type:%d vlen:%d, val:%" PRIi8 "\n", tag, ln, (int32_t)type, vlen, *(int8_t *)val);
      break;
C
Cary Xu 已提交
787 788 789 790 791 792 793
    default:
      ASSERT(0);
      break;
  }
}

void debugPrintSTag(STag *pTag, const char *tag, int32_t ln) {
H
Hongze Cheng 已提交
794 795
  int8_t   isJson = pTag->flags & TD_TAG_JSON;
  int8_t   isLarge = pTag->flags & TD_TAG_LARGE;
796 797 798 799 800 801 802 803 804 805
  uint8_t *p = NULL;
  int16_t  offset = 0;

  if (isLarge) {
    p = (uint8_t *)&((int16_t *)pTag->idx)[pTag->nTag];
  } else {
    p = (uint8_t *)&pTag->idx[pTag->nTag];
  }
  printf("%s:%d >>> STAG === %s:%s, len: %d, nTag: %d, sver:%d\n", tag, ln, isJson ? "json" : "normal",
         isLarge ? "large" : "small", (int32_t)pTag->len, (int32_t)pTag->nTag, pTag->ver);
C
Cary Xu 已提交
806
  for (uint16_t n = 0; n < pTag->nTag; ++n) {
807 808 809 810 811 812 813 814
    if (isLarge) {
      offset = ((int16_t *)pTag->idx)[n];
    } else {
      offset = pTag->idx[n];
    }
    STagVal tagVal = {0};
    if (isJson) {
      tagVal.pKey = (char *)POINTER_SHIFT(p, offset);
C
Cary Xu 已提交
815
    } else {
816
      tagVal.cid = *(int16_t *)POINTER_SHIFT(p, offset);
C
Cary Xu 已提交
817
    }
818
    printf("%s:%d loop[%d-%d] offset=%d\n", __func__, __LINE__, (int32_t)pTag->nTag, (int32_t)n, (int32_t)offset);
C
Cary Xu 已提交
819
    tGetTagVal(p + offset, &tagVal, isJson);
820
    if (IS_VAR_DATA_TYPE(tagVal.type)) {
wmmhello's avatar
wmmhello 已提交
821
      debugPrintTagVal(tagVal.type, tagVal.pData, tagVal.nData, __func__, __LINE__);
822
    } else {
wmmhello's avatar
wmmhello 已提交
823 824
      debugPrintTagVal(tagVal.type, &tagVal.i64, tDataTypes[tagVal.type].bytes, __func__, __LINE__);
    }
C
Cary Xu 已提交
825 826 827 828
  }
  printf("\n");
}

H
Hongze Cheng 已提交
829 830
static int32_t tPutTagVal(uint8_t *p, STagVal *pTagVal, int8_t isJson) {
  int32_t n = 0;
H
Hongze Cheng 已提交
831

H
Hongze Cheng 已提交
832 833 834 835 836 837
  // key
  if (isJson) {
    n += tPutCStr(p ? p + n : p, pTagVal->pKey);
  } else {
    n += tPutI16v(p ? p + n : p, pTagVal->cid);
  }
H
Hongze Cheng 已提交
838

H
Hongze Cheng 已提交
839 840 841 842 843 844 845
  // type
  n += tPutI8(p ? p + n : p, pTagVal->type);

  // value
  if (IS_VAR_DATA_TYPE(pTagVal->type)) {
    n += tPutBinary(p ? p + n : p, pTagVal->pData, pTagVal->nData);
  } else {
H
Hongze Cheng 已提交
846
    p = p ? p + n : p;
wmmhello's avatar
wmmhello 已提交
847
    n += tDataTypes[pTagVal->type].bytes;
848
    if (p) memcpy(p, &(pTagVal->i64), tDataTypes[pTagVal->type].bytes);
H
Hongze Cheng 已提交
849 850
  }

H
Hongze Cheng 已提交
851 852 853 854 855 856 857 858 859 860
  return n;
}
static int32_t tGetTagVal(uint8_t *p, STagVal *pTagVal, int8_t isJson) {
  int32_t n = 0;

  // key
  if (isJson) {
    n += tGetCStr(p + n, &pTagVal->pKey);
  } else {
    n += tGetI16v(p + n, &pTagVal->cid);
H
Hongze Cheng 已提交
861 862
  }

H
Hongze Cheng 已提交
863 864
  // type
  n += tGetI8(p + n, &pTagVal->type);
H
Hongze Cheng 已提交
865

H
Hongze Cheng 已提交
866 867 868 869
  // value
  if (IS_VAR_DATA_TYPE(pTagVal->type)) {
    n += tGetBinary(p + n, &pTagVal->pData, &pTagVal->nData);
  } else {
wmmhello's avatar
wmmhello 已提交
870
    memcpy(&(pTagVal->i64), p + n, tDataTypes[pTagVal->type].bytes);
wmmhello's avatar
wmmhello 已提交
871
    n += tDataTypes[pTagVal->type].bytes;
H
Hongze Cheng 已提交
872 873 874 875
  }

  return n;
}
wmmhello's avatar
wmmhello 已提交
876

H
Hongze Cheng 已提交
877
bool tTagIsJson(const void *pTag) { return (((const STag *)pTag)->flags & TD_TAG_JSON); }
wmmhello's avatar
wmmhello 已提交
878

H
Hongze Cheng 已提交
879 880 881 882 883
bool tTagIsJsonNull(void *data) {
  STag  *pTag = (STag *)data;
  int8_t isJson = tTagIsJson(pTag);
  if (!isJson) return false;
  return ((STag *)data)->nTag == 0;
wmmhello's avatar
wmmhello 已提交
884 885
}

H
Hongze Cheng 已提交
886
int32_t tTagNew(SArray *pArray, int32_t version, int8_t isJson, STag **ppTag) {
H
Hongze Cheng 已提交
887 888 889
  int32_t  code = 0;
  uint8_t *p = NULL;
  int16_t  n = 0;
H
Hongze Cheng 已提交
890
  int16_t  nTag = taosArrayGetSize(pArray);
H
Hongze Cheng 已提交
891 892
  int32_t  szTag = 0;
  int8_t   isLarge = 0;
H
Hongze Cheng 已提交
893 894 895

  // sort
  if (isJson) {
wafwerar's avatar
wafwerar 已提交
896
    taosSort(pArray->pData, nTag, sizeof(STagVal), tTagValJsonCmprFn);
H
Hongze Cheng 已提交
897
  } else {
wafwerar's avatar
wafwerar 已提交
898
    taosSort(pArray->pData, nTag, sizeof(STagVal), tTagValCmprFn);
H
Hongze Cheng 已提交
899 900 901
  }

  // get size
H
Hongze Cheng 已提交
902
  for (int16_t iTag = 0; iTag < nTag; iTag++) {
H
Hongze Cheng 已提交
903
    szTag += tPutTagVal(NULL, (STagVal *)taosArrayGet(pArray, iTag), isJson);
H
Hongze Cheng 已提交
904
  }
H
Hongze Cheng 已提交
905 906 907 908 909 910
  if (szTag <= INT8_MAX) {
    szTag = szTag + sizeof(STag) + sizeof(int8_t) * nTag;
  } else {
    szTag = szTag + sizeof(STag) + sizeof(int16_t) * nTag;
    isLarge = 1;
  }
H
Hongze Cheng 已提交
911

H
Hongze Cheng 已提交
912
  ASSERT(szTag <= INT16_MAX);
H
Hongze Cheng 已提交
913 914

  // build tag
C
Cary Xu 已提交
915
  (*ppTag) = (STag *)taosMemoryCalloc(szTag, 1);
H
Hongze Cheng 已提交
916 917 918 919
  if ((*ppTag) == NULL) {
    code = TSDB_CODE_OUT_OF_MEMORY;
    goto _err;
  }
H
Hongze Cheng 已提交
920 921 922 923 924 925 926
  (*ppTag)->flags = 0;
  if (isJson) {
    (*ppTag)->flags |= TD_TAG_JSON;
  }
  if (isLarge) {
    (*ppTag)->flags |= TD_TAG_LARGE;
  }
H
Hongze Cheng 已提交
927 928 929
  (*ppTag)->len = szTag;
  (*ppTag)->nTag = nTag;
  (*ppTag)->ver = version;
H
Hongze Cheng 已提交
930

H
Hongze Cheng 已提交
931 932 933 934 935
  if (isLarge) {
    p = (uint8_t *)&((int16_t *)(*ppTag)->idx)[nTag];
  } else {
    p = (uint8_t *)&(*ppTag)->idx[nTag];
  }
H
Hongze Cheng 已提交
936 937
  n = 0;
  for (int16_t iTag = 0; iTag < nTag; iTag++) {
H
Hongze Cheng 已提交
938 939 940 941 942
    if (isLarge) {
      ((int16_t *)(*ppTag)->idx)[iTag] = n;
    } else {
      (*ppTag)->idx[iTag] = n;
    }
H
Hongze Cheng 已提交
943
    n += tPutTagVal(p + n, (STagVal *)taosArrayGet(pArray, iTag), isJson);
H
Hongze Cheng 已提交
944
  }
945
#ifdef TD_DEBUG_PRINT_TAG
C
Cary Xu 已提交
946
  debugPrintSTag(*ppTag, __func__, __LINE__);
947
#endif
C
Cary Xu 已提交
948

H
Hongze Cheng 已提交
949 950 951 952
  return code;

_err:
  return code;
H
Hongze Cheng 已提交
953 954 955 956 957 958
}

void tTagFree(STag *pTag) {
  if (pTag) taosMemoryFree(pTag);
}

959 960 961
char *tTagValToData(const STagVal *value, bool isJson) {
  if (!value) return NULL;
  char  *data = NULL;
wmmhello's avatar
wmmhello 已提交
962 963 964 965
  int8_t typeBytes = 0;
  if (isJson) {
    typeBytes = CHAR_BYTES;
  }
966
  if (IS_VAR_DATA_TYPE(value->type)) {
wmmhello's avatar
wmmhello 已提交
967
    data = taosMemoryCalloc(1, typeBytes + VARSTR_HEADER_SIZE + value->nData);
968 969
    if (data == NULL) return NULL;
    if (isJson) *data = value->type;
wmmhello's avatar
wmmhello 已提交
970 971
    varDataLen(data + typeBytes) = value->nData;
    memcpy(varDataVal(data + typeBytes), value->pData, value->nData);
972 973
  } else {
    data = ((char *)&(value->i64)) - typeBytes;  // json with type
wmmhello's avatar
wmmhello 已提交
974 975 976 977 978
  }

  return data;
}

C
Cary Xu 已提交
979
bool tTagGet(const STag *pTag, STagVal *pTagVal) {
H
Hongze Cheng 已提交
980 981 982
  int16_t  lidx = 0;
  int16_t  ridx = pTag->nTag - 1;
  int16_t  midx;
H
Hongze Cheng 已提交
983 984 985 986
  uint8_t *p;
  int8_t   isJson = pTag->flags & TD_TAG_JSON;
  int8_t   isLarge = pTag->flags & TD_TAG_LARGE;
  int16_t  offset;
H
Hongze Cheng 已提交
987 988 989
  STagVal  tv;
  int      c;

H
Hongze Cheng 已提交
990 991 992 993 994 995
  if (isLarge) {
    p = (uint8_t *)&((int16_t *)pTag->idx)[pTag->nTag];
  } else {
    p = (uint8_t *)&pTag->idx[pTag->nTag];
  }

H
Hongze Cheng 已提交
996 997 998 999 1000
  pTagVal->type = TSDB_DATA_TYPE_NULL;
  pTagVal->pData = NULL;
  pTagVal->nData = 0;
  while (lidx <= ridx) {
    midx = (lidx + ridx) / 2;
H
Hongze Cheng 已提交
1001 1002 1003 1004 1005
    if (isLarge) {
      offset = ((int16_t *)pTag->idx)[midx];
    } else {
      offset = pTag->idx[midx];
    }
H
Hongze Cheng 已提交
1006

H
Hongze Cheng 已提交
1007 1008
    tGetTagVal(p + offset, &tv, isJson);
    if (isJson) {
H
Hongze Cheng 已提交
1009
      c = tTagValJsonCmprFn(pTagVal, &tv);
H
Hongze Cheng 已提交
1010
    } else {
H
Hongze Cheng 已提交
1011
      c = tTagValCmprFn(pTagVal, &tv);
H
Hongze Cheng 已提交
1012 1013
    }

H
Hongze Cheng 已提交
1014 1015 1016 1017
    if (c < 0) {
      ridx = midx - 1;
    } else if (c > 0) {
      lidx = midx + 1;
H
Hongze Cheng 已提交
1018
    } else {
H
Hongze Cheng 已提交
1019
      memcpy(pTagVal, &tv, sizeof(tv));
C
Cary Xu 已提交
1020
      return true;
H
Hongze Cheng 已提交
1021 1022
    }
  }
C
Cary Xu 已提交
1023
  return false;
H
Hongze Cheng 已提交
1024 1025
}

H
more  
Hongze Cheng 已提交
1026 1027
int32_t tEncodeTag(SEncoder *pEncoder, const STag *pTag) {
  return tEncodeBinary(pEncoder, (const uint8_t *)pTag, pTag->len);
H
Hongze Cheng 已提交
1028 1029
}

H
more  
Hongze Cheng 已提交
1030
int32_t tDecodeTag(SDecoder *pDecoder, STag **ppTag) { return tDecodeBinary(pDecoder, (uint8_t **)ppTag, NULL); }
H
Hongze Cheng 已提交
1031

C
Cary Xu 已提交
1032
int32_t tTagToValArray(const STag *pTag, SArray **ppArray) {
H
Hongze Cheng 已提交
1033
  int32_t  code = 0;
1034 1035 1036 1037 1038 1039 1040 1041 1042 1043
  uint8_t *p = NULL;
  STagVal  tv = {0};
  int8_t   isLarge = pTag->flags & TD_TAG_LARGE;
  int16_t  offset = 0;

  if (isLarge) {
    p = (uint8_t *)&((int16_t *)pTag->idx)[pTag->nTag];
  } else {
    p = (uint8_t *)&pTag->idx[pTag->nTag];
  }
H
Hongze Cheng 已提交
1044

H
more  
Hongze Cheng 已提交
1045
  (*ppArray) = taosArrayInit(pTag->nTag + 1, sizeof(STagVal));
H
Hongze Cheng 已提交
1046
  if (*ppArray == NULL) {
H
Hongze Cheng 已提交
1047 1048 1049 1050 1051
    code = TSDB_CODE_OUT_OF_MEMORY;
    goto _err;
  }

  for (int16_t iTag = 0; iTag < pTag->nTag; iTag++) {
1052 1053 1054 1055 1056 1057
    if (isLarge) {
      offset = ((int16_t *)pTag->idx)[iTag];
    } else {
      offset = pTag->idx[iTag];
    }
    tGetTagVal(p + offset, &tv, pTag->flags & TD_TAG_JSON);
H
Hongze Cheng 已提交
1058
    taosArrayPush(*ppArray, &tv);
H
Hongze Cheng 已提交
1059 1060 1061 1062 1063 1064 1065 1066
  }

  return code;

_err:
  return code;
}

H
Hongze Cheng 已提交
1067
#if 1  // ===================================================================================================================
C
Cary Xu 已提交
1068
int tdInitTSchemaBuilder(STSchemaBuilder *pBuilder, schema_ver_t version) {
H
Hongze Cheng 已提交
1069 1070 1071
  if (pBuilder == NULL) return -1;

  pBuilder->tCols = 256;
wafwerar's avatar
wafwerar 已提交
1072
  pBuilder->columns = (STColumn *)taosMemoryMalloc(sizeof(STColumn) * pBuilder->tCols);
H
Hongze Cheng 已提交
1073 1074 1075 1076 1077 1078 1079 1080
  if (pBuilder->columns == NULL) return -1;

  tdResetTSchemaBuilder(pBuilder, version);
  return 0;
}

void tdDestroyTSchemaBuilder(STSchemaBuilder *pBuilder) {
  if (pBuilder) {
wafwerar's avatar
wafwerar 已提交
1081
    taosMemoryFreeClear(pBuilder->columns);
H
Hongze Cheng 已提交
1082 1083 1084
  }
}

C
Cary Xu 已提交
1085
void tdResetTSchemaBuilder(STSchemaBuilder *pBuilder, schema_ver_t version) {
H
Hongze Cheng 已提交
1086 1087 1088
  pBuilder->nCols = 0;
  pBuilder->tlen = 0;
  pBuilder->flen = 0;
T
Tao Liu 已提交
1089
  pBuilder->vlen = 0;
H
Hongze Cheng 已提交
1090 1091 1092
  pBuilder->version = version;
}

C
Cary Xu 已提交
1093
int32_t tdAddColToSchema(STSchemaBuilder *pBuilder, int8_t type, int8_t flags, col_id_t colId, col_bytes_t bytes) {
1094
  if (!isValidDataType(type)) return -1;
H
Hongze Cheng 已提交
1095 1096 1097

  if (pBuilder->nCols >= pBuilder->tCols) {
    pBuilder->tCols *= 2;
wafwerar's avatar
wafwerar 已提交
1098
    STColumn *columns = (STColumn *)taosMemoryRealloc(pBuilder->columns, sizeof(STColumn) * pBuilder->tCols);
T
tickduan 已提交
1099 1100
    if (columns == NULL) return -1;
    pBuilder->columns = columns;
H
Hongze Cheng 已提交
1101 1102 1103
  }

  STColumn *pCol = &(pBuilder->columns[pBuilder->nCols]);
H
Hongze Cheng 已提交
1104 1105 1106
  pCol->type = type;
  pCol->colId = colId;
  pCol->flags = flags;
H
Hongze Cheng 已提交
1107
  if (pBuilder->nCols == 0) {
H
Hongze Cheng 已提交
1108
    pCol->offset = 0;
H
Hongze Cheng 已提交
1109
  } else {
S
Shengliang Guan 已提交
1110
    STColumn *pTCol = &(pBuilder->columns[pBuilder->nCols - 1]);
H
Hongze Cheng 已提交
1111
    pCol->offset = pTCol->offset + TYPE_BYTES[pTCol->type];
H
Hongze Cheng 已提交
1112 1113 1114
  }

  if (IS_VAR_DATA_TYPE(type)) {
H
Hongze Cheng 已提交
1115
    pCol->bytes = bytes;
T
Tao Liu 已提交
1116 1117
    pBuilder->tlen += (TYPE_BYTES[type] + bytes);
    pBuilder->vlen += bytes - sizeof(VarDataLenT);
H
Hongze Cheng 已提交
1118
  } else {
H
Hongze Cheng 已提交
1119
    pCol->bytes = TYPE_BYTES[type];
H
Hongze Cheng 已提交
1120
    pBuilder->tlen += TYPE_BYTES[type];
T
Tao Liu 已提交
1121
    pBuilder->vlen += TYPE_BYTES[type];
H
Hongze Cheng 已提交
1122 1123 1124 1125 1126 1127 1128 1129 1130 1131 1132 1133 1134 1135 1136
  }

  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 已提交
1137
  STSchema *pSchema = (STSchema *)taosMemoryMalloc(tlen);
H
Hongze Cheng 已提交
1138 1139 1140 1141 1142 1143
  if (pSchema == NULL) return NULL;

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

C
Cary Xu 已提交
1146
#ifdef TD_SUPPORT_BITMAP
C
Cary Xu 已提交
1147
  schemaTLen(pSchema) += (int)TD_BITMAP_BYTES(schemaNCols(pSchema));
C
Cary Xu 已提交
1148 1149
#endif

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

H
TD-27  
hzcheng 已提交
1152 1153 1154
  return pSchema;
}

H
Hongze Cheng 已提交
1155
#endif