scalar.c 52.9 KB
Newer Older
H
Hongze Cheng 已提交
1
#include "scalar.h"
D
dapan1121 已提交
2 3
#include "function.h"
#include "functionMgt.h"
H
Haojun Liao 已提交
4 5
#include "nodes.h"
#include "querynodes.h"
D
dapan1121 已提交
6
#include "sclInt.h"
H
Haojun Liao 已提交
7 8
#include "sclvector.h"
#include "tcommon.h"
H
Hongze Cheng 已提交
9
#include "tcompare.h"
H
Haojun Liao 已提交
10
#include "tdatablock.h"
D
fix bug  
dapan1121 已提交
11
#include "ttime.h"
H
Hongze Cheng 已提交
12
#include "tudf.h"
D
dapan1121 已提交
13

D
dapan1121 已提交
14
int32_t scalarGetOperatorParamNum(EOperatorType type) {
H
Hongze Cheng 已提交
15 16 17
  if (OP_TYPE_IS_NULL == type || OP_TYPE_IS_NOT_NULL == type || OP_TYPE_IS_TRUE == type ||
      OP_TYPE_IS_NOT_TRUE == type || OP_TYPE_IS_FALSE == type || OP_TYPE_IS_NOT_FALSE == type ||
      OP_TYPE_IS_UNKNOWN == type || OP_TYPE_IS_NOT_UNKNOWN == type || OP_TYPE_MINUS == type) {
D
dapan1121 已提交
18 19 20 21 22 23
    return 1;
  }

  return 2;
}

H
Hongze Cheng 已提交
24 25
int32_t sclConvertToTsValueNode(int8_t precision, SValueNode *valueNode) {
  char   *timeStr = valueNode->datum.p;
X
Xiaoyu Wang 已提交
26 27
  int64_t value = 0;
  int32_t code = convertStringToTimestamp(valueNode->node.resType.type, valueNode->datum.p, precision, &value);
D
dapan1121 已提交
28 29
  if (code != TSDB_CODE_SUCCESS) {
    return code;
D
dapan 已提交
30 31
  }
  taosMemoryFree(timeStr);
X
Xiaoyu Wang 已提交
32
  valueNode->datum.i = value;
D
dapan1121 已提交
33
  valueNode->typeData = valueNode->datum.i;
G
Ganlin Zhao 已提交
34

D
dapan 已提交
35 36
  valueNode->node.resType.type = TSDB_DATA_TYPE_TIMESTAMP;
  valueNode->node.resType.bytes = tDataTypes[TSDB_DATA_TYPE_TIMESTAMP].bytes;
D
dapan1121 已提交
37 38

  return TSDB_CODE_SUCCESS;
D
dapan 已提交
39 40
}

H
Hongze Cheng 已提交
41 42
int32_t sclCreateColumnInfoData(SDataType *pType, int32_t numOfRows, SScalarParam *pParam) {
  SColumnInfoData *pColumnData = taosMemoryCalloc(1, sizeof(SColumnInfoData));
43 44
  if (pColumnData == NULL) {
    terrno = TSDB_CODE_OUT_OF_MEMORY;
H
Haojun Liao 已提交
45
    return terrno;
46 47
  }

H
Hongze Cheng 已提交
48 49 50
  pColumnData->info.type = pType->type;
  pColumnData->info.bytes = pType->bytes;
  pColumnData->info.scale = pType->scale;
51 52
  pColumnData->info.precision = pType->precision;

H
Haojun Liao 已提交
53
  int32_t code = colInfoDataEnsureCapacity(pColumnData, numOfRows, true);
54 55
  if (code != TSDB_CODE_SUCCESS) {
    terrno = TSDB_CODE_OUT_OF_MEMORY;
H
Haojun Liao 已提交
56
    taosMemoryFree(pColumnData);
H
Haojun Liao 已提交
57
    return terrno;
58
  }
H
Haojun Liao 已提交
59 60

  pParam->columnData = pColumnData;
D
dapan1121 已提交
61
  pParam->colAlloced = true;
62 63
  pParam->numOfRows = numOfRows;

H
Haojun Liao 已提交
64
  return TSDB_CODE_SUCCESS;
65 66
}

X
Xiaoyu Wang 已提交
67
int32_t sclConvertValueToSclParam(SValueNode *pValueNode, SScalarParam *out, int32_t *overflow) {
68
  SScalarParam in = {.numOfRows = 1};
H
Hongze Cheng 已提交
69
  int32_t      code = sclCreateColumnInfoData(&pValueNode->node.resType, 1, &in);
H
Haojun Liao 已提交
70 71 72 73
  if (code != TSDB_CODE_SUCCESS) {
    return code;
  }

74
  colDataSetVal(in.columnData, 0, nodesGetValueFromNode(pValueNode), false);
75

H
Haojun Liao 已提交
76
  colInfoDataEnsureCapacity(out->columnData, 1, true);
D
dapan1121 已提交
77
  code = vectorConvertSingleColImpl(&in, out, overflow, -1, -1);
78 79 80
  sclFreeParam(&in);

  return code;
81 82
}

D
dapan1121 已提交
83
int32_t sclExtendResRows(SScalarParam *pDst, SScalarParam *pSrc, SArray *pBlockList) {
X
Xiaoyu Wang 已提交
84
  SSDataBlock  *pb = taosArrayGetP(pBlockList, 0);
D
dapan1121 已提交
85 86 87
  SScalarParam *pLeft = taosMemoryCalloc(1, sizeof(SScalarParam));
  if (NULL == pLeft) {
    sclError("calloc %d failed", (int32_t)sizeof(SScalarParam));
S
Shengliang Guan 已提交
88
    SCL_ERR_RET(TSDB_CODE_OUT_OF_MEMORY);
D
dapan1121 已提交
89 90 91 92 93
  }

  pLeft->numOfRows = pb->info.rows;

  if (pDst->numOfRows < pb->info.rows) {
H
Haojun Liao 已提交
94
    colInfoDataEnsureCapacity(pDst->columnData, pb->info.rows, true);
D
dapan1121 已提交
95
  }
X
Xiaoyu Wang 已提交
96

D
dapan1121 已提交
97 98 99 100 101 102 103 104
  _bin_scalar_fn_t OperatorFn = getBinScalarOperatorFn(OP_TYPE_ASSIGN);
  OperatorFn(pLeft, pSrc, pDst, TSDB_ORDER_ASC);

  taosMemoryFree(pLeft);

  return TSDB_CODE_SUCCESS;
}

D
dapan1121 已提交
105 106 107 108
int32_t scalarGenerateSetFromList(void **data, void *pNode, uint32_t type) {
  SHashObj *pObj = taosHashInit(256, taosGetDefaultHashFunction(type), true, false);
  if (NULL == pObj) {
    sclError("taosHashInit failed, size:%d", 256);
S
Shengliang Guan 已提交
109
    SCL_ERR_RET(TSDB_CODE_OUT_OF_MEMORY);
D
dapan1121 已提交
110 111
  }

G
Ganlin Zhao 已提交
112
  taosHashSetEqualFp(pObj, taosGetDefaultEqualFunction(type));
D
dapan1121 已提交
113

H
Hongze Cheng 已提交
114
  int32_t        code = 0;
D
dapan1121 已提交
115
  SNodeListNode *nodeList = (SNodeListNode *)pNode;
H
Hongze Cheng 已提交
116 117
  SListCell     *cell = nodeList->pNodeList->pHead;
  SScalarParam   out = {.columnData = taosMemoryCalloc(1, sizeof(SColumnInfoData))};
118

D
dapan1121 已提交
119
  int32_t len = 0;
H
Hongze Cheng 已提交
120
  void   *buf = NULL;
G
Ganlin Zhao 已提交
121

D
dapan1121 已提交
122 123
  for (int32_t i = 0; i < nodeList->pNodeList->length; ++i) {
    SValueNode *valueNode = (SValueNode *)cell->pNode;
G
Ganlin Zhao 已提交
124

D
dapan1121 已提交
125
    if (valueNode->node.resType.type != type) {
126
      out.columnData->info.type = type;
D
dapan1121 已提交
127 128 129 130 131 132 133 134 135
      if (IS_VAR_DATA_TYPE(type)) {
        if (IS_VAR_DATA_TYPE(valueNode->node.resType.type)) {
          out.columnData->info.bytes = valueNode->node.resType.bytes * TSDB_NCHAR_SIZE;
        } else {
          out.columnData->info.bytes = 64 * TSDB_NCHAR_SIZE;
        }
      } else {
        out.columnData->info.bytes = tDataTypes[type].bytes;
      }
136

D
dapan1121 已提交
137
      int32_t overflow = 0;
D
dapan1121 已提交
138
      code = sclConvertValueToSclParam(valueNode, &out, &overflow);
139
      if (code != TSDB_CODE_SUCCESS) {
H
Hongze Cheng 已提交
140
        //        sclError("convert data from %d to %d failed", in.type, out.type);
D
dapan1121 已提交
141 142 143
        SCL_ERR_JRET(code);
      }

D
dapan1121 已提交
144 145 146 147 148
      if (overflow) {
        cell = cell->pNext;
        continue;
      }

D
dapan1121 已提交
149
      if (IS_VAR_DATA_TYPE(type)) {
150
        buf = colDataGetVarData(out.columnData, 0);
D
dapan1121 已提交
151
        len = varDataTLen(buf);
D
dapan1121 已提交
152 153
      } else {
        len = tDataTypes[type].bytes;
154
        buf = out.columnData->pData;
D
dapan1121 已提交
155 156 157
      }
    } else {
      buf = nodesGetValueFromNode(valueNode);
D
dapan1121 已提交
158
      if (IS_VAR_DATA_TYPE(type)) {
159
        len = varDataTLen(buf);
D
dapan1121 已提交
160 161
      } else {
        len = valueNode->node.resType.bytes;
162
      }
D
dapan1121 已提交
163
    }
G
Ganlin Zhao 已提交
164

165
    if (taosHashPut(pObj, buf, (size_t)len, NULL, 0)) {
D
dapan1121 已提交
166
      sclError("taosHashPut to set failed");
S
Shengliang Guan 已提交
167
      SCL_ERR_JRET(TSDB_CODE_OUT_OF_MEMORY);
D
dapan1121 已提交
168
    }
D
dapan1121 已提交
169

170
    colInfoDataCleanup(out.columnData, out.numOfRows);
D
dapan1121 已提交
171
    cell = cell->pNext;
D
dapan1121 已提交
172 173 174
  }

  *data = pObj;
D
dapan1121 已提交
175 176 177

  colDataDestroy(out.columnData);
  taosMemoryFreeClear(out.columnData);
D
dapan1121 已提交
178 179 180
  return TSDB_CODE_SUCCESS;

_return:
D
dapan1121 已提交
181 182 183

  colDataDestroy(out.columnData);
  taosMemoryFreeClear(out.columnData);
D
dapan1121 已提交
184 185 186 187
  taosHashCleanup(pObj);
  SCL_RET(code);
}

D
dapan1121 已提交
188 189
void sclFreeRes(SHashObj *res) {
  SScalarParam *p = NULL;
H
Hongze Cheng 已提交
190
  void         *pIter = taosHashIterate(res, NULL);
D
dapan1121 已提交
191 192 193 194
  while (pIter) {
    p = (SScalarParam *)pIter;

    if (p) {
D
dapan 已提交
195
      sclFreeParam(p);
D
dapan1121 已提交
196 197 198 199 200 201
    }
    pIter = taosHashIterate(res, pIter);
  }
  taosHashCleanup(res);
}

D
dapan1121 已提交
202
void sclFreeParam(SScalarParam *param) {
D
dapan1121 已提交
203
  if (NULL == param || !param->colAlloced) {
D
dapan1121 已提交
204 205
    return;
  }
H
Hongze Cheng 已提交
206

D
dapan1121 已提交
207
  if (param->columnData != NULL) {
208
    colDataDestroy(param->columnData);
209
    taosMemoryFreeClear(param->columnData);
210 211 212 213
  }

  if (param->pHashFilter != NULL) {
    taosHashCleanup(param->pHashFilter);
D
dapan1121 已提交
214
    param->pHashFilter = NULL;
215
  }
D
dapan1121 已提交
216 217
}

D
dapan1121 已提交
218 219 220 221
int32_t sclCopyValueNodeValue(SValueNode *pNode, void **res) {
  if (TSDB_DATA_TYPE_NULL == pNode->node.resType.type) {
    return TSDB_CODE_SUCCESS;
  }
G
Ganlin Zhao 已提交
222

wafwerar's avatar
wafwerar 已提交
223
  *res = taosMemoryMalloc(pNode->node.resType.bytes);
D
dapan1121 已提交
224 225
  if (NULL == (*res)) {
    sclError("malloc %d failed", pNode->node.resType.bytes);
S
Shengliang Guan 已提交
226
    SCL_ERR_RET(TSDB_CODE_OUT_OF_MEMORY);
D
dapan1121 已提交
227 228 229 230 231 232
  }

  memcpy(*res, nodesGetValueFromNode(pNode), pNode->node.resType.bytes);
  return TSDB_CODE_SUCCESS;
}

D
dapan1121 已提交
233 234 235 236 237 238
void sclFreeParamList(SScalarParam *param, int32_t paramNum) {
  if (NULL == param) {
    return;
  }

  for (int32_t i = 0; i < paramNum; ++i) {
H
Hongze Cheng 已提交
239
    SScalarParam *p = param + i;
D
dapan1121 已提交
240 241 242 243 244 245
    sclFreeParam(p);
  }

  taosMemoryFree(param);
}

246 247 248 249 250 251
void sclDowngradeValueType(SValueNode *valueNode) {
  switch (valueNode->node.resType.type) {
    case TSDB_DATA_TYPE_BIGINT: {
      int8_t i8 = valueNode->datum.i;
      if (i8 == valueNode->datum.i) {
        valueNode->node.resType.type = TSDB_DATA_TYPE_TINYINT;
H
Hongze Cheng 已提交
252
        *(int8_t *)&valueNode->typeData = i8;
253 254 255 256 257
        break;
      }
      int16_t i16 = valueNode->datum.i;
      if (i16 == valueNode->datum.i) {
        valueNode->node.resType.type = TSDB_DATA_TYPE_SMALLINT;
H
Hongze Cheng 已提交
258
        *(int16_t *)&valueNode->typeData = i16;
259 260 261 262 263
        break;
      }
      int32_t i32 = valueNode->datum.i;
      if (i32 == valueNode->datum.i) {
        valueNode->node.resType.type = TSDB_DATA_TYPE_INT;
H
Hongze Cheng 已提交
264
        *(int32_t *)&valueNode->typeData = i32;
265 266 267 268
        break;
      }
      break;
    }
H
Hongze Cheng 已提交
269
    case TSDB_DATA_TYPE_UBIGINT: {
270 271 272 273 274
      uint8_t u8 = valueNode->datum.i;
      if (u8 == valueNode->datum.i) {
        int8_t i8 = valueNode->datum.i;
        if (i8 == valueNode->datum.i) {
          valueNode->node.resType.type = TSDB_DATA_TYPE_TINYINT;
H
Hongze Cheng 已提交
275
          *(int8_t *)&valueNode->typeData = i8;
276 277
        } else {
          valueNode->node.resType.type = TSDB_DATA_TYPE_UTINYINT;
H
Hongze Cheng 已提交
278
          *(uint8_t *)&valueNode->typeData = u8;
279 280 281 282 283 284 285 286
        }
        break;
      }
      uint16_t u16 = valueNode->datum.i;
      if (u16 == valueNode->datum.i) {
        int16_t i16 = valueNode->datum.i;
        if (i16 == valueNode->datum.i) {
          valueNode->node.resType.type = TSDB_DATA_TYPE_SMALLINT;
H
Hongze Cheng 已提交
287
          *(int16_t *)&valueNode->typeData = i16;
288 289
        } else {
          valueNode->node.resType.type = TSDB_DATA_TYPE_USMALLINT;
H
Hongze Cheng 已提交
290
          *(uint16_t *)&valueNode->typeData = u16;
291 292 293 294 295 296 297 298
        }
        break;
      }
      uint32_t u32 = valueNode->datum.i;
      if (u32 == valueNode->datum.i) {
        int32_t i32 = valueNode->datum.i;
        if (i32 == valueNode->datum.i) {
          valueNode->node.resType.type = TSDB_DATA_TYPE_INT;
H
Hongze Cheng 已提交
299
          *(int32_t *)&valueNode->typeData = i32;
300 301
        } else {
          valueNode->node.resType.type = TSDB_DATA_TYPE_UINT;
H
Hongze Cheng 已提交
302
          *(uint32_t *)&valueNode->typeData = u32;
303 304 305 306 307 308 309 310 311
        }
        break;
      }
      break;
    }
    case TSDB_DATA_TYPE_DOUBLE: {
      float f = valueNode->datum.d;
      if (FLT_EQUAL(f, valueNode->datum.d)) {
        valueNode->node.resType.type = TSDB_DATA_TYPE_FLOAT;
H
Hongze Cheng 已提交
312
        *(float *)&valueNode->typeData = f;
313 314 315 316 317 318 319 320 321
        break;
      }
      break;
    }
    default:
      break;
  }
}

H
Hongze Cheng 已提交
322
int32_t sclInitParam(SNode *node, SScalarParam *param, SScalarCtx *ctx, int32_t *rowNum) {
D
dapan1121 已提交
323
  switch (nodeType(node)) {
D
dapan1121 已提交
324
    case QUERY_NODE_LEFT_VALUE: {
H
Hongze Cheng 已提交
325
      SSDataBlock *pb = taosArrayGetP(ctx->pBlockList, 0);
D
dapan1121 已提交
326 327 328
      param->numOfRows = pb->info.rows;
      break;
    }
D
dapan1121 已提交
329 330
    case QUERY_NODE_VALUE: {
      SValueNode *valueNode = (SValueNode *)node;
331

G
Ganlin Zhao 已提交
332 333 334 335
      if (param->columnData != NULL) {
        sclError("columnData should be NULL");
        SCL_ERR_RET(TSDB_CODE_QRY_INVALID_INPUT);
      }
336
      param->numOfRows = 1;
G
Ganlin Zhao 已提交
337 338 339 340
      int32_t code = sclCreateColumnInfoData(&valueNode->node.resType, 1, param);
      if (code != TSDB_CODE_SUCCESS) {
        SCL_RET(TSDB_CODE_OUT_OF_MEMORY);
      }
wmmhello's avatar
wmmhello 已提交
341
      if (TSDB_DATA_TYPE_NULL == valueNode->node.resType.type || valueNode->isNull) {
342
        colDataSetNULL(param->columnData, 0);
343
      } else {
344
        colDataSetVal(param->columnData, 0, nodesGetValueFromNode(valueNode), false);
D
dapan1121 已提交
345
      }
D
dapan1121 已提交
346 347
      break;
    }
D
dapan1121 已提交
348 349
    case QUERY_NODE_NODE_LIST: {
      SNodeListNode *nodeList = (SNodeListNode *)node;
350 351
      if (LIST_LENGTH(nodeList->pNodeList) <= 0) {
        sclError("invalid length in nodeList, length:%d", LIST_LENGTH(nodeList->pNodeList));
D
dapan1121 已提交
352 353 354
        SCL_RET(TSDB_CODE_QRY_INVALID_INPUT);
      }

D
dapan1121 已提交
355 356
      int32_t type = vectorGetConvertType(ctx->type.selfType, ctx->type.peerType);
      if (type == 0) {
357
        type = nodeList->node.resType.type;
D
dapan1121 已提交
358
      }
G
Ganlin Zhao 已提交
359

D
dapan1121 已提交
360 361
      SCL_ERR_RET(scalarGenerateSetFromList((void **)&param->pHashFilter, node, type));
      param->hashValueType = type;
D
dapan1121 已提交
362
      param->colAlloced = true;
D
dapan 已提交
363
      if (taosHashPut(ctx->pRes, &node, POINTER_BYTES, param, sizeof(*param))) {
364
        taosHashCleanup(param->pHashFilter);
D
dapan1121 已提交
365
        param->pHashFilter = NULL;
D
dapan 已提交
366
        sclError("taosHashPut nodeList failed, size:%d", (int32_t)sizeof(*param));
S
Shengliang Guan 已提交
367
        return TSDB_CODE_OUT_OF_MEMORY;
G
Ganlin Zhao 已提交
368
      }
D
dapan1121 已提交
369
      param->colAlloced = false;
D
dapan1121 已提交
370 371
      break;
    }
X
Xiaoyu Wang 已提交
372
    case QUERY_NODE_COLUMN: {
D
dapan 已提交
373 374
      if (NULL == ctx->pBlockList) {
        sclError("invalid node type for constant calculating, type:%d, src:%p", nodeType(node), ctx->pBlockList);
S
Shengliang Guan 已提交
375
        SCL_ERR_RET(TSDB_CODE_APP_ERROR);
D
dapan1121 已提交
376
      }
G
Ganlin Zhao 已提交
377

X
Xiaoyu Wang 已提交
378
      SColumnNode *ref = (SColumnNode *)node;
379 380

      int32_t index = -1;
H
Hongze Cheng 已提交
381 382
      for (int32_t i = 0; i < taosArrayGetSize(ctx->pBlockList); ++i) {
        SSDataBlock *pb = taosArrayGetP(ctx->pBlockList, i);
H
Haojun Liao 已提交
383
        if (pb->info.id.blockId == ref->dataBlockId) {
384 385 386 387 388 389
          index = i;
          break;
        }
      }

      if (index == -1) {
H
Hongze Cheng 已提交
390 391
        sclError("column tupleId is too big, tupleId:%d, dataBlockNum:%d", ref->dataBlockId,
                 (int32_t)taosArrayGetSize(ctx->pBlockList));
D
dapan 已提交
392 393 394
        SCL_ERR_RET(TSDB_CODE_QRY_INVALID_INPUT);
      }

395
      SSDataBlock *block = *(SSDataBlock **)taosArrayGet(ctx->pBlockList, index);
G
Ganlin Zhao 已提交
396 397 398 399 400
      if (NULL == block) {
        SCL_ERR_RET(TSDB_CODE_QRY_INVALID_INPUT);
      }

      if (ref->slotId >= taosArrayGetSize(block->pDataBlock)) {
H
Hongze Cheng 已提交
401 402
        sclError("column slotId is too big, slodId:%d, dataBlockSize:%d", ref->slotId,
                 (int32_t)taosArrayGetSize(block->pDataBlock));
D
dapan1121 已提交
403 404 405
        SCL_ERR_RET(TSDB_CODE_QRY_INVALID_INPUT);
      }

D
dapan 已提交
406
      SColumnInfoData *columnData = (SColumnInfoData *)taosArrayGet(block->pDataBlock, ref->slotId);
407
#if TAG_FILTER_DEBUG
H
Hongze Cheng 已提交
408 409
      qDebug("tagfilter column info, slotId:%d, colId:%d, type:%d", ref->slotId, columnData->info.colId,
             columnData->info.type);
410
#endif
411 412
      param->numOfRows = block->info.rows;
      param->columnData = columnData;
D
dapan1121 已提交
413 414
      break;
    }
415 416
    case QUERY_NODE_FUNCTION:
    case QUERY_NODE_OPERATOR:
D
dapan1121 已提交
417 418
    case QUERY_NODE_LOGIC_CONDITION:
    case QUERY_NODE_CASE_WHEN: {
D
dapan1121 已提交
419 420 421
      SScalarParam *res = (SScalarParam *)taosHashGet(ctx->pRes, &node, POINTER_BYTES);
      if (NULL == res) {
        sclError("no result for node, type:%d, node:%p", nodeType(node), node);
S
Shengliang Guan 已提交
422
        SCL_ERR_RET(TSDB_CODE_APP_ERROR);
D
dapan1121 已提交
423 424
      }
      *param = *res;
D
dapan1121 已提交
425
      param->colAlloced = false;
D
dapan1121 已提交
426 427
      break;
    }
428 429
    default:
      break;
D
dapan1121 已提交
430 431
  }

432 433 434
  if (param->numOfRows > *rowNum) {
    if ((1 != param->numOfRows) && (1 < *rowNum)) {
      sclError("different row nums, rowNum:%d, newRowNum:%d", *rowNum, param->numOfRows);
D
dapan1121 已提交
435 436
      SCL_ERR_RET(TSDB_CODE_QRY_INVALID_INPUT);
    }
G
Ganlin Zhao 已提交
437

438
    *rowNum = param->numOfRows;
D
dapan1121 已提交
439 440
  }

441
  param->param = ctx->param;
D
dapan1121 已提交
442 443 444
  return TSDB_CODE_SUCCESS;
}

H
Hongze Cheng 已提交
445 446
int32_t sclInitParamList(SScalarParam **pParams, SNodeList *pParamList, SScalarCtx *ctx, int32_t *paramNum,
                         int32_t *rowNum) {
D
dapan1121 已提交
447
  int32_t code = 0;
D
dapan1121 已提交
448 449
  if (NULL == pParamList) {
    if (ctx->pBlockList) {
D
dapan1121 已提交
450
      SSDataBlock *pBlock = taosArrayGetP(ctx->pBlockList, 0);
D
dapan1121 已提交
451 452 453 454 455
      *rowNum = pBlock->info.rows;
    } else {
      *rowNum = 1;
    }

D
dapan 已提交
456
    *paramNum = 1;
D
dapan1121 已提交
457
  } else {
D
dapan 已提交
458
    *paramNum = pParamList->length;
D
dapan1121 已提交
459 460
  }

D
dapan 已提交
461
  SScalarParam *paramList = taosMemoryCalloc(*paramNum, sizeof(SScalarParam));
D
dapan1121 已提交
462
  if (NULL == paramList) {
D
dapan 已提交
463
    sclError("calloc %d failed", (int32_t)((*paramNum) * sizeof(SScalarParam)));
S
Shengliang Guan 已提交
464
    SCL_ERR_RET(TSDB_CODE_OUT_OF_MEMORY);
D
dapan1121 已提交
465
  }
D
dapan1121 已提交
466
  if (pParamList) {
H
Hongze Cheng 已提交
467
    SNode  *tnode = NULL;
D
dapan1121 已提交
468 469
    int32_t i = 0;
    if (SCL_IS_CONST_CALC(ctx)) {
H
Hongze Cheng 已提交
470
      WHERE_EACH(tnode, pParamList) {
D
dapan1121 已提交
471
        if (!SCL_IS_CONST_NODE(tnode)) {
D
dapan 已提交
472
          WHERE_NEXT;
D
dapan1121 已提交
473 474 475 476
        } else {
          SCL_ERR_JRET(sclInitParam(tnode, &paramList[i], ctx, rowNum));
          ERASE_NODE(pParamList);
        }
G
Ganlin Zhao 已提交
477

D
dapan1121 已提交
478 479 480
        ++i;
      }
    } else {
G
Ganlin Zhao 已提交
481
      FOREACH(tnode, pParamList) {
D
dapan1121 已提交
482 483 484
        SCL_ERR_JRET(sclInitParam(tnode, &paramList[i], ctx, rowNum));
        ++i;
      }
D
dapan1121 已提交
485
    }
D
dapan1121 已提交
486 487 488
  } else {
    paramList[0].numOfRows = *rowNum;
  }
D
dapan1121 已提交
489

D
dapan1121 已提交
490
  if (0 == *rowNum) {
G
Ganlin Zhao 已提交
491
    taosMemoryFreeClear(paramList);
D
dapan1121 已提交
492
  }
D
dapan1121 已提交
493

D
dapan1121 已提交
494
  *pParams = paramList;
D
dapan1121 已提交
495
  return TSDB_CODE_SUCCESS;
D
dapan1121 已提交
496

D
dapan1121 已提交
497
_return:
wafwerar's avatar
wafwerar 已提交
498
  taosMemoryFreeClear(paramList);
D
dapan1121 已提交
499 500 501
  SCL_RET(code);
}

D
dapan1121 已提交
502 503 504 505
int32_t sclGetNodeType(SNode *pNode, SScalarCtx *ctx) {
  if (NULL == pNode) {
    return -1;
  }
G
Ganlin Zhao 已提交
506

wafwerar's avatar
wafwerar 已提交
507
  switch ((int)nodeType(pNode)) {
D
dapan1121 已提交
508 509 510 511 512 513
    case QUERY_NODE_VALUE: {
      SValueNode *valueNode = (SValueNode *)pNode;
      return valueNode->node.resType.type;
    }
    case QUERY_NODE_NODE_LIST: {
      SNodeListNode *nodeList = (SNodeListNode *)pNode;
514
      return nodeList->node.resType.type;
D
dapan1121 已提交
515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540
    }
    case QUERY_NODE_COLUMN: {
      SColumnNode *colNode = (SColumnNode *)pNode;
      return colNode->node.resType.type;
    }
    case QUERY_NODE_FUNCTION:
    case QUERY_NODE_OPERATOR:
    case QUERY_NODE_LOGIC_CONDITION: {
      SScalarParam *res = (SScalarParam *)taosHashGet(ctx->pRes, &pNode, POINTER_BYTES);
      if (NULL == res) {
        sclError("no result for node, type:%d, node:%p", nodeType(pNode), pNode);
        return -1;
      }
      return res->columnData->info.type;
    }
  }

  return -1;
}

void sclSetOperatorValueType(SOperatorNode *node, SScalarCtx *ctx) {
  ctx->type.opResType = node->node.resType.type;
  ctx->type.selfType = sclGetNodeType(node->pLeft, ctx);
  ctx->type.peerType = sclGetNodeType(node->pRight, ctx);
}

D
dapan1121 已提交
541 542
int32_t sclInitOperatorParams(SScalarParam **pParams, SOperatorNode *node, SScalarCtx *ctx, int32_t *rowNum) {
  int32_t code = 0;
D
dapan1121 已提交
543
  int32_t paramNum = scalarGetOperatorParamNum(node->opType);
D
dapan1121 已提交
544 545 546 547
  if (NULL == node->pLeft || (paramNum == 2 && NULL == node->pRight)) {
    sclError("invalid operation node, left:%p, right:%p", node->pLeft, node->pRight);
    SCL_ERR_RET(TSDB_CODE_QRY_INVALID_INPUT);
  }
G
Ganlin Zhao 已提交
548

wafwerar's avatar
wafwerar 已提交
549
  SScalarParam *paramList = taosMemoryCalloc(paramNum, sizeof(SScalarParam));
D
dapan1121 已提交
550 551
  if (NULL == paramList) {
    sclError("calloc %d failed", (int32_t)(paramNum * sizeof(SScalarParam)));
S
Shengliang Guan 已提交
552
    SCL_ERR_RET(TSDB_CODE_OUT_OF_MEMORY);
D
dapan1121 已提交
553 554
  }

D
dapan1121 已提交
555 556
  sclSetOperatorValueType(node, ctx);

D
dapan1121 已提交
557
  SCL_ERR_JRET(sclInitParam(node->pLeft, &paramList[0], ctx, rowNum));
D
dapan1121 已提交
558
  if (paramNum > 1) {
D
dapan1121 已提交
559
    TSWAP(ctx->type.selfType, ctx->type.peerType);
D
dapan1121 已提交
560
    SCL_ERR_JRET(sclInitParam(node->pRight, &paramList[1], ctx, rowNum));
D
dapan1121 已提交
561 562
  }

D
dapan1121 已提交
563
  *pParams = paramList;
D
dapan1121 已提交
564
  return TSDB_CODE_SUCCESS;
D
dapan1121 已提交
565 566

_return:
wafwerar's avatar
wafwerar 已提交
567
  taosMemoryFreeClear(paramList);
D
dapan1121 已提交
568
  SCL_RET(code);
D
dapan1121 已提交
569 570
}

X
Xiaoyu Wang 已提交
571
int32_t sclGetNodeRes(SNode *node, SScalarCtx *ctx, SScalarParam **res) {
D
dapan1121 已提交
572 573 574 575 576 577 578 579 580
  if (NULL == node) {
    return TSDB_CODE_SUCCESS;
  }

  int32_t rowNum = 0;
  *res = taosMemoryCalloc(1, sizeof(**res));
  if (NULL == *res) {
    SCL_ERR_RET(TSDB_CODE_OUT_OF_MEMORY);
  }
X
Xiaoyu Wang 已提交
581

D
dapan1121 已提交
582 583 584 585 586
  SCL_ERR_RET(sclInitParam(node, *res, ctx, &rowNum));

  return TSDB_CODE_SUCCESS;
}

X
Xiaoyu Wang 已提交
587 588 589 590 591 592 593 594 595 596 597 598
int32_t sclWalkCaseWhenList(SScalarCtx *ctx, SNodeList *pList, struct SListCell *pCell, SScalarParam *pCase,
                            SScalarParam *pElse, SScalarParam *pComp, SScalarParam *output, int32_t rowIdx,
                            int32_t totalRows, bool *complete) {
  SNode         *node = NULL;
  SWhenThenNode *pWhenThen = NULL;
  SScalarParam  *pWhen = NULL;
  SScalarParam  *pThen = NULL;
  int32_t        code = 0;

  for (SListCell *cell = pCell; (NULL != cell ? (node = cell->pNode, true) : (node = NULL, false));
       cell = cell->pNext) {
    pWhenThen = (SWhenThenNode *)node;
D
dapan1121 已提交
599 600 601

    SCL_ERR_RET(sclGetNodeRes(pWhenThen->pWhen, ctx, &pWhen));
    SCL_ERR_RET(sclGetNodeRes(pWhenThen->pThen, ctx, &pThen));
X
Xiaoyu Wang 已提交
602

D
dapan1121 已提交
603
    vectorCompareImpl(pCase, pWhen, pComp, rowIdx, 1, TSDB_ORDER_ASC, OP_TYPE_EQUAL);
X
Xiaoyu Wang 已提交
604 605

    bool *equal = (bool *)colDataGetData(pComp->columnData, rowIdx);
D
dapan1121 已提交
606
    if (*equal) {
dengyihao's avatar
dengyihao 已提交
607
      bool  isNull = colDataIsNull_s(pThen->columnData, (pThen->numOfRows > 1 ? rowIdx : 0));
D
dapan1121 已提交
608
      char *pData = isNull ? NULL : colDataGetData(pThen->columnData, (pThen->numOfRows > 1 ? rowIdx : 0));
609
      colDataSetVal(output->columnData, rowIdx, pData, isNull);
D
dapan1121 已提交
610 611 612 613 614

      if (0 == rowIdx && 1 == pCase->numOfRows && 1 == pWhen->numOfRows && 1 == pThen->numOfRows && totalRows > 1) {
        SCL_ERR_JRET(sclExtendResRows(output, output, ctx->pBlockList));
        *complete = true;
      }
X
Xiaoyu Wang 已提交
615

D
dapan1121 已提交
616
      goto _return;
D
dapan1121 已提交
617 618 619 620
    }
  }

  if (pElse) {
dengyihao's avatar
dengyihao 已提交
621
    bool  isNull = colDataIsNull_s(pElse->columnData, (pElse->numOfRows > 1 ? rowIdx : 0));
D
dapan1121 已提交
622
    char *pData = isNull ? NULL : colDataGetData(pElse->columnData, (pElse->numOfRows > 1 ? rowIdx : 0));
623
    colDataSetVal(output->columnData, rowIdx, pData, isNull);
D
dapan1121 已提交
624 625 626 627 628

    if (0 == rowIdx && 1 == pCase->numOfRows && 1 == pElse->numOfRows && totalRows > 1) {
      SCL_ERR_JRET(sclExtendResRows(output, output, ctx->pBlockList));
      *complete = true;
    }
X
Xiaoyu Wang 已提交
629

D
dapan1121 已提交
630
    goto _return;
D
dapan1121 已提交
631 632
  }

633
  colDataSetVal(output->columnData, rowIdx, NULL, true);
D
dapan1121 已提交
634

D
dapan1121 已提交
635
  if (0 == rowIdx && 1 == pCase->numOfRows && totalRows > 1) {
D
dapan1121 已提交
636 637 638 639 640
    SCL_ERR_JRET(sclExtendResRows(output, output, ctx->pBlockList));
    *complete = true;
  }

_return:
X
Xiaoyu Wang 已提交
641

D
dapan1121 已提交
642 643
  sclFreeParam(pWhen);
  sclFreeParam(pThen);
D
dapan1121 已提交
644 645
  taosMemoryFree(pWhen);
  taosMemoryFree(pThen);
D
dapan1121 已提交
646 647

  SCL_RET(code);
D
dapan1121 已提交
648 649
}

X
Xiaoyu Wang 已提交
650 651 652 653 654 655 656
int32_t sclWalkWhenList(SScalarCtx *ctx, SNodeList *pList, struct SListCell *pCell, SScalarParam *pElse,
                        SScalarParam *output, int32_t rowIdx, int32_t totalRows, bool *complete, bool preSingle) {
  SNode         *node = NULL;
  SWhenThenNode *pWhenThen = NULL;
  SScalarParam  *pWhen = NULL;
  SScalarParam  *pThen = NULL;
  int32_t        code = 0;
D
dapan1121 已提交
657

X
Xiaoyu Wang 已提交
658 659 660
  for (SListCell *cell = pCell; (NULL != cell ? (node = cell->pNode, true) : (node = NULL, false));
       cell = cell->pNext) {
    pWhenThen = (SWhenThenNode *)node;
D
dapan1121 已提交
661 662
    pWhen = NULL;
    pThen = NULL;
X
Xiaoyu Wang 已提交
663

D
dapan1121 已提交
664 665 666
    SCL_ERR_JRET(sclGetNodeRes(pWhenThen->pWhen, ctx, &pWhen));
    SCL_ERR_JRET(sclGetNodeRes(pWhenThen->pThen, ctx, &pThen));

X
Xiaoyu Wang 已提交
667 668
    bool *whenValue = (bool *)colDataGetData(pWhen->columnData, (pWhen->numOfRows > 1 ? rowIdx : 0));

D
dapan1121 已提交
669
    if (*whenValue) {
dengyihao's avatar
dengyihao 已提交
670
      bool  isNull = colDataIsNull_s(pThen->columnData, (pThen->numOfRows > 1 ? rowIdx : 0));
D
dapan1121 已提交
671
      char *pData = isNull ? NULL : colDataGetData(pThen->columnData, (pThen->numOfRows > 1 ? rowIdx : 0));
672
      colDataSetVal(output->columnData, rowIdx, pData, isNull);
D
dapan1121 已提交
673

D
dapan1121 已提交
674
      if (preSingle && 0 == rowIdx && 1 == pWhen->numOfRows && 1 == pThen->numOfRows && totalRows > 1) {
D
dapan1121 已提交
675
        SCL_ERR_JRET(sclExtendResRows(output, output, ctx->pBlockList));
D
dapan1121 已提交
676
        *complete = true;
D
dapan1121 已提交
677
      }
X
Xiaoyu Wang 已提交
678

D
dapan1121 已提交
679 680 681 682 683
      goto _return;
    }

    sclFreeParam(pWhen);
    sclFreeParam(pThen);
D
dapan1121 已提交
684 685
    taosMemoryFreeClear(pWhen);
    taosMemoryFreeClear(pThen);
D
dapan1121 已提交
686 687 688
  }

  if (pElse) {
dengyihao's avatar
dengyihao 已提交
689
    bool  isNull = colDataIsNull_s(pElse->columnData, (pElse->numOfRows > 1 ? rowIdx : 0));
D
dapan1121 已提交
690
    char *pData = isNull ? NULL : colDataGetData(pElse->columnData, (pElse->numOfRows > 1 ? rowIdx : 0));
691
    colDataSetVal(output->columnData, rowIdx, pData, isNull);
D
dapan1121 已提交
692

D
dapan1121 已提交
693
    if (preSingle && 0 == rowIdx && 1 == pElse->numOfRows && totalRows > 1) {
D
dapan1121 已提交
694
      SCL_ERR_JRET(sclExtendResRows(output, output, ctx->pBlockList));
D
dapan1121 已提交
695
      *complete = true;
D
dapan1121 已提交
696
    }
X
Xiaoyu Wang 已提交
697

D
dapan1121 已提交
698 699 700
    goto _return;
  }

701
  colDataSetVal(output->columnData, rowIdx, NULL, true);
D
dapan1121 已提交
702

D
dapan1121 已提交
703
  if (preSingle && 0 == rowIdx && totalRows > 1) {
D
dapan1121 已提交
704
    SCL_ERR_JRET(sclExtendResRows(output, output, ctx->pBlockList));
D
dapan1121 已提交
705
    *complete = true;
D
dapan1121 已提交
706 707
  }

D
dapan1121 已提交
708 709 710 711
_return:

  sclFreeParam(pWhen);
  sclFreeParam(pThen);
D
dapan1121 已提交
712 713
  taosMemoryFree(pWhen);
  taosMemoryFree(pThen);
D
dapan1121 已提交
714

D
dapan1121 已提交
715
  SCL_RET(code);
D
dapan1121 已提交
716 717
}

718
int32_t sclExecFunction(SFunctionNode *node, SScalarCtx *ctx, SScalarParam *output) {
D
dapan1121 已提交
719
  SScalarParam *params = NULL;
H
Hongze Cheng 已提交
720 721 722
  int32_t       rowNum = 0;
  int32_t       paramNum = 0;
  int32_t       code = 0;
D
dapan 已提交
723
  SCL_ERR_RET(sclInitParamList(&params, node->pParameterList, ctx, &paramNum, &rowNum));
D
dapan1121 已提交
724

D
dapan1121 已提交
725
  if (fmIsUserDefinedFunc(node->funcId)) {
726
    code = callUdfScalarFunc(node->functionName, params, paramNum, output);
727 728 729 730
    if (code != 0) {
      sclError("fmExecFunction error. callUdfScalarFunc. function name: %s, udf code:%d", node->functionName, code);
      goto _return;
    }
D
dapan1121 已提交
731 732 733 734 735 736 737
  } else {
    SScalarFuncExecFuncs ffpSet = {0};
    code = fmGetScalarFuncExecFuncs(node->funcId, &ffpSet);
    if (code) {
      sclError("fmGetFuncExecFuncs failed, funcId:%d, code:%s", node->funcId, tstrerror(code));
      SCL_ERR_JRET(code);
    }
G
Ganlin Zhao 已提交
738

H
Haojun Liao 已提交
739 740 741
    code = sclCreateColumnInfoData(&node->node.resType, rowNum, output);
    if (code != TSDB_CODE_SUCCESS) {
      SCL_ERR_JRET(code);
D
dapan1121 已提交
742 743
    }

744 745 746 747
    if (rowNum == 0) {
      goto _return;
    }

D
dapan1121 已提交
748 749 750 751 752
    code = (*ffpSet.process)(params, paramNum, output);
    if (code) {
      sclError("scalar function exec failed, funcId:%d, code:%s", node->funcId, tstrerror(code));
      SCL_ERR_JRET(code);
    }
D
dapan1121 已提交
753 754 755 756
  }

_return:

D
dapan1121 已提交
757
  sclFreeParamList(params, paramNum);
D
dapan1121 已提交
758 759 760 761 762
  SCL_RET(code);
}

int32_t sclExecLogic(SLogicConditionNode *node, SScalarCtx *ctx, SScalarParam *output) {
  if (NULL == node->pParameterList || node->pParameterList->length <= 0) {
H
Hongze Cheng 已提交
763 764
    sclError("invalid logic parameter list, list:%p, paramNum:%d", node->pParameterList,
             node->pParameterList ? node->pParameterList->length : 0);
D
dapan1121 已提交
765 766 767 768 769 770
    SCL_ERR_RET(TSDB_CODE_QRY_INVALID_INPUT);
  }

  if (TSDB_DATA_TYPE_BOOL != node->node.resType.type) {
    sclError("invalid logic resType, type:%d", node->node.resType.type);
    SCL_ERR_RET(TSDB_CODE_QRY_INVALID_INPUT);
D
dapan1121 已提交
771 772
  }

D
dapan1121 已提交
773 774 775 776 777 778
  if (LOGIC_COND_TYPE_NOT == node->condType && node->pParameterList->length > 1) {
    sclError("invalid NOT operation parameter number, paramNum:%d", node->pParameterList->length);
    SCL_ERR_RET(TSDB_CODE_QRY_INVALID_INPUT);
  }

  SScalarParam *params = NULL;
H
Hongze Cheng 已提交
779 780 781
  int32_t       rowNum = 0;
  int32_t       paramNum = 0;
  int32_t       code = 0;
D
dapan 已提交
782
  SCL_ERR_RET(sclInitParamList(&params, node->pParameterList, ctx, &paramNum, &rowNum));
D
dapan1121 已提交
783 784 785 786
  if (NULL == params) {
    output->numOfRows = 0;
    return TSDB_CODE_SUCCESS;
  }
D
dapan1121 已提交
787

788 789 790 791
  int32_t type = node->node.resType.type;
  output->numOfRows = rowNum;

  SDataType t = {.type = type, .bytes = tDataTypes[type].bytes};
H
Haojun Liao 已提交
792 793 794
  code = sclCreateColumnInfoData(&t, rowNum, output);
  if (code != TSDB_CODE_SUCCESS) {
    SCL_ERR_JRET(code);
D
dapan1121 已提交
795
  }
D
dapan1121 已提交
796

797 798
  int32_t numOfQualified = 0;

D
dapan1121 已提交
799
  bool value = false;
D
dapan 已提交
800
  bool complete = true;
D
dapan1121 已提交
801
  for (int32_t i = 0; i < rowNum; ++i) {
D
dapan 已提交
802 803
    complete = true;
    for (int32_t m = 0; m < paramNum; ++m) {
D
dapan1121 已提交
804
      if (NULL == params[m].columnData) {
D
dapan 已提交
805
        complete = false;
D
dapan1121 已提交
806 807
        continue;
      }
H
Hongze Cheng 已提交
808
      char *p = colDataGetData(params[m].columnData, i);
809 810
      GET_TYPED_DATA(value, bool, params[m].columnData->info.type, p);

D
dapan1121 已提交
811
      if (LOGIC_COND_TYPE_AND == node->condType && (false == value)) {
D
dapan1121 已提交
812
        complete = true;
D
dapan1121 已提交
813 814
        break;
      } else if (LOGIC_COND_TYPE_OR == node->condType && value) {
D
dapan1121 已提交
815
        complete = true;
D
dapan1121 已提交
816 817 818 819 820 821
        break;
      } else if (LOGIC_COND_TYPE_NOT == node->condType) {
        value = !value;
      }
    }

D
dapan 已提交
822
    if (complete) {
823
      colDataSetVal(output->columnData, i, (char *)&value, false);
824 825 826
      if (value) {
        numOfQualified++;
      }
D
dapan 已提交
827
    }
D
dapan1121 已提交
828 829
  }

D
dapan1121 已提交
830 831 832 833 834
  if (SCL_IS_CONST_CALC(ctx) && (false == complete)) {
    sclFreeParam(output);
    output->numOfRows = 0;
  }

835
  output->numOfQualified = numOfQualified;
D
dapan1121 已提交
836

837
_return:
D
dapan1121 已提交
838
  sclFreeParamList(params, paramNum);
D
dapan1121 已提交
839
  SCL_RET(code);
D
dapan1121 已提交
840 841 842 843
}

int32_t sclExecOperator(SOperatorNode *node, SScalarCtx *ctx, SScalarParam *output) {
  SScalarParam *params = NULL;
H
Hongze Cheng 已提交
844 845
  int32_t       rowNum = 0;
  int32_t       code = 0;
846

wmmhello's avatar
wmmhello 已提交
847
  // json not support in in operator
H
Haojun Liao 已提交
848
  if (nodeType(node->pLeft) == QUERY_NODE_VALUE) {
wmmhello's avatar
wmmhello 已提交
849
    SValueNode *valueNode = (SValueNode *)node->pLeft;
H
Hongze Cheng 已提交
850 851
    if (valueNode->node.resType.type == TSDB_DATA_TYPE_JSON &&
        (node->opType == OP_TYPE_IN || node->opType == OP_TYPE_NOT_IN)) {
wmmhello's avatar
wmmhello 已提交
852 853 854 855
      SCL_RET(TSDB_CODE_QRY_JSON_IN_ERROR);
    }
  }

D
dapan1121 已提交
856
  SCL_ERR_RET(sclInitOperatorParams(&params, node, ctx, &rowNum));
857
  if (output->columnData == NULL) {
H
Haojun Liao 已提交
858 859 860 861
    code = sclCreateColumnInfoData(&node->node.resType, rowNum, output);
    if (code != TSDB_CODE_SUCCESS) {
      SCL_ERR_JRET(code);
    }
D
dapan1121 已提交
862 863 864 865
  }

  _bin_scalar_fn_t OperatorFn = getBinScalarOperatorFn(node->opType);

H
Hongze Cheng 已提交
866 867 868
  int32_t       paramNum = scalarGetOperatorParamNum(node->opType);
  SScalarParam *pLeft = &params[0];
  SScalarParam *pRight = paramNum > 1 ? &params[1] : NULL;
869

wmmhello's avatar
wmmhello 已提交
870
  terrno = TSDB_CODE_SUCCESS;
D
dapan 已提交
871
  OperatorFn(pLeft, pRight, output, TSDB_ORDER_ASC);
wmmhello's avatar
wmmhello 已提交
872
  code = terrno;
D
dapan1121 已提交
873 874

_return:
D
dapan1121 已提交
875

D
dapan1121 已提交
876
  sclFreeParamList(params, paramNum);
D
dapan1121 已提交
877
  SCL_RET(code);
D
dapan1121 已提交
878 879
}

D
dapan1121 已提交
880
int32_t sclExecCaseWhen(SCaseWhenNode *node, SScalarCtx *ctx, SScalarParam *output) {
X
Xiaoyu Wang 已提交
881
  int32_t       code = 0;
D
dapan1121 已提交
882 883
  SScalarParam *pCase = NULL;
  SScalarParam *pElse = NULL;
D
dapan1121 已提交
884 885
  SScalarParam *pWhen = NULL;
  SScalarParam *pThen = NULL;
D
dapan1121 已提交
886
  SScalarParam  comp = {0};
X
Xiaoyu Wang 已提交
887 888
  int32_t       rowNum = 1;
  bool          complete = false;
D
dapan1121 已提交
889 890 891 892 893 894 895

  if (NULL == node->pWhenThenList || node->pWhenThenList->length <= 0) {
    sclError("invalid whenThen list");
    SCL_ERR_RET(TSDB_CODE_INVALID_PARA);
  }

  if (ctx->pBlockList) {
X
Xiaoyu Wang 已提交
896
    SSDataBlock *pb = taosArrayGetP(ctx->pBlockList, 0);
D
dapan1121 已提交
897
    rowNum = pb->info.rows;
D
dapan1121 已提交
898
    output->numOfRows = pb->info.rows;
D
dapan1121 已提交
899
  }
D
dapan1121 已提交
900 901

  SCL_ERR_JRET(sclCreateColumnInfoData(&node->node.resType, rowNum, output));
X
Xiaoyu Wang 已提交
902

D
dapan1121 已提交
903 904 905 906 907 908
  SCL_ERR_JRET(sclGetNodeRes(node->pCase, ctx, &pCase));
  SCL_ERR_JRET(sclGetNodeRes(node->pElse, ctx, &pElse));

  SDataType compType = {0};
  compType.type = TSDB_DATA_TYPE_BOOL;
  compType.bytes = tDataTypes[compType.type].bytes;
X
Xiaoyu Wang 已提交
909

D
dapan1121 已提交
910
  SCL_ERR_JRET(sclCreateColumnInfoData(&compType, rowNum, &comp));
D
dapan1121 已提交
911

X
Xiaoyu Wang 已提交
912 913
  SNode         *tnode = NULL;
  SWhenThenNode *pWhenThen = (SWhenThenNode *)node->pWhenThenList->pHead->pNode;
D
dapan1121 已提交
914 915 916

  SCL_ERR_JRET(sclGetNodeRes(pWhenThen->pWhen, ctx, &pWhen));
  SCL_ERR_JRET(sclGetNodeRes(pWhenThen->pThen, ctx, &pThen));
D
dapan1121 已提交
917 918 919 920
  if (NULL == pWhen || NULL == pThen) {
    sclError("invalid when/then in whenThen list");
    SCL_ERR_JRET(TSDB_CODE_INVALID_PARA);
  }
D
dapan1121 已提交
921 922

  if (pCase) {
D
dapan1121 已提交
923
    vectorCompare(pCase, pWhen, &comp, TSDB_ORDER_ASC, OP_TYPE_EQUAL);
X
Xiaoyu Wang 已提交
924

D
dapan1121 已提交
925
    for (int32_t i = 0; i < rowNum; ++i) {
X
Xiaoyu Wang 已提交
926
      bool *equal = (bool *)colDataGetData(comp.columnData, (comp.numOfRows > 1 ? i : 0));
D
dapan1121 已提交
927
      if (*equal) {
928
        colDataSetVal(output->columnData, i, colDataGetData(pThen->columnData, (pThen->numOfRows > 1 ? i : 0)),
X
Xiaoyu Wang 已提交
929
                      colDataIsNull_s(pThen->columnData, (pThen->numOfRows > 1 ? i : 0)));
D
dapan1121 已提交
930
        if (0 == i && 1 == pCase->numOfRows && 1 == pWhen->numOfRows && 1 == pThen->numOfRows && rowNum > 1) {
D
dapan1121 已提交
931 932 933
          SCL_ERR_JRET(sclExtendResRows(output, output, ctx->pBlockList));
          break;
        }
D
dapan1121 已提交
934
      } else {
X
Xiaoyu Wang 已提交
935 936
        SCL_ERR_JRET(sclWalkCaseWhenList(ctx, node->pWhenThenList, node->pWhenThenList->pHead->pNext, pCase, pElse,
                                         &comp, output, i, rowNum, &complete));
D
dapan1121 已提交
937 938 939
        if (complete) {
          break;
        }
D
dapan1121 已提交
940
      }
D
dapan1121 已提交
941
    }
D
dapan1121 已提交
942 943
  } else {
    for (int32_t i = 0; i < rowNum; ++i) {
X
Xiaoyu Wang 已提交
944
      bool *whenValue = (bool *)colDataGetData(pWhen->columnData, (pWhen->numOfRows > 1 ? i : 0));
D
dapan1121 已提交
945
      if (*whenValue) {
946
        colDataSetVal(output->columnData, i, colDataGetData(pThen->columnData, (pThen->numOfRows > 1 ? i : 0)),
X
Xiaoyu Wang 已提交
947
                      colDataIsNull_s(pThen->columnData, (pThen->numOfRows > 1 ? i : 0)));
D
dapan1121 已提交
948 949 950 951
        if (0 == i && 1 == pWhen->numOfRows && 1 == pThen->numOfRows && rowNum > 1) {
          SCL_ERR_JRET(sclExtendResRows(output, output, ctx->pBlockList));
          break;
        }
D
dapan1121 已提交
952
      } else {
X
Xiaoyu Wang 已提交
953 954
        SCL_ERR_JRET(sclWalkWhenList(ctx, node->pWhenThenList, node->pWhenThenList->pHead->pNext, pElse, output, i,
                                     rowNum, &complete, (pWhen->numOfRows == 1 && pThen->numOfRows == 1)));
D
dapan1121 已提交
955 956 957
        if (complete) {
          break;
        }
D
dapan1121 已提交
958
      }
D
dapan1121 已提交
959 960 961
    }
  }

D
dapan1121 已提交
962 963
  sclFreeParam(pCase);
  sclFreeParam(pElse);
D
dapan1121 已提交
964
  sclFreeParam(&comp);
D
dapan1121 已提交
965 966
  sclFreeParam(pWhen);
  sclFreeParam(pThen);
D
dapan1121 已提交
967 968 969 970
  taosMemoryFree(pCase);
  taosMemoryFree(pElse);
  taosMemoryFree(pWhen);
  taosMemoryFree(pThen);
D
dapan1121 已提交
971

D
dapan1121 已提交
972
  return TSDB_CODE_SUCCESS;
D
dapan1121 已提交
973 974 975

_return:

D
dapan1121 已提交
976 977
  sclFreeParam(pCase);
  sclFreeParam(pElse);
D
dapan1121 已提交
978
  sclFreeParam(&comp);
D
dapan1121 已提交
979 980 981
  sclFreeParam(pWhen);
  sclFreeParam(pThen);
  sclFreeParam(output);
D
dapan1121 已提交
982 983 984 985
  taosMemoryFree(pCase);
  taosMemoryFree(pElse);
  taosMemoryFree(pWhen);
  taosMemoryFree(pThen);
D
dapan1121 已提交
986

D
dapan1121 已提交
987 988 989
  SCL_RET(code);
}

H
Hongze Cheng 已提交
990
EDealRes sclRewriteNullInOptr(SNode **pNode, SScalarCtx *ctx, EOperatorType opType) {
D
dapan1121 已提交
991 992 993
  if (opType <= OP_TYPE_CALC_MAX) {
    SValueNode *res = (SValueNode *)nodesMakeNode(QUERY_NODE_VALUE);
    if (NULL == res) {
G
Ganlin Zhao 已提交
994
      sclError("make value node failed");
S
Shengliang Guan 已提交
995
      ctx->code = TSDB_CODE_OUT_OF_MEMORY;
D
dapan1121 已提交
996 997
      return DEAL_RES_ERROR;
    }
G
Ganlin Zhao 已提交
998

D
dapan1121 已提交
999
    res->node.resType.type = TSDB_DATA_TYPE_NULL;
G
Ganlin Zhao 已提交
1000

D
dapan1121 已提交
1001
    nodesDestroyNode(*pNode);
H
Hongze Cheng 已提交
1002
    *pNode = (SNode *)res;
D
dapan1121 已提交
1003 1004 1005
  } else {
    SValueNode *res = (SValueNode *)nodesMakeNode(QUERY_NODE_VALUE);
    if (NULL == res) {
G
Ganlin Zhao 已提交
1006
      sclError("make value node failed");
S
Shengliang Guan 已提交
1007
      ctx->code = TSDB_CODE_OUT_OF_MEMORY;
D
dapan1121 已提交
1008 1009
      return DEAL_RES_ERROR;
    }
G
Ganlin Zhao 已提交
1010

D
dapan1121 已提交
1011
    res->node.resType.type = TSDB_DATA_TYPE_BOOL;
D
dapan1121 已提交
1012
    res->node.resType.bytes = tDataTypes[TSDB_DATA_TYPE_BOOL].bytes;
D
dapan1121 已提交
1013
    res->datum.b = false;
G
Ganlin Zhao 已提交
1014

D
dapan1121 已提交
1015
    nodesDestroyNode(*pNode);
H
Hongze Cheng 已提交
1016
    *pNode = (SNode *)res;
D
dapan1121 已提交
1017 1018 1019 1020 1021
  }

  return DEAL_RES_CONTINUE;
}

H
Hongze Cheng 已提交
1022
EDealRes sclAggFuncWalker(SNode *pNode, void *pContext) {
D
dapan1121 已提交
1023
  if (QUERY_NODE_FUNCTION == nodeType(pNode)) {
H
Hongze Cheng 已提交
1024 1025 1026
    SFunctionNode *pFunc = (SFunctionNode *)pNode;
    *(bool *)pContext = fmIsAggFunc(pFunc->funcId);
    if (*(bool *)pContext) {
D
dapan1121 已提交
1027 1028 1029 1030 1031 1032 1033
      return DEAL_RES_END;
    }
  }

  return DEAL_RES_CONTINUE;
}

H
Hongze Cheng 已提交
1034
bool sclContainsAggFuncNode(SNode *pNode) {
D
dapan1121 已提交
1035 1036 1037 1038
  bool aggFunc = false;
  nodesWalkExpr(pNode, sclAggFuncWalker, (void *)&aggFunc);
  return aggFunc;
}
D
dapan1121 已提交
1039

1040 1041 1042 1043

int32_t sclConvertOpValueNodeTs(SOperatorNode *node, SScalarCtx *ctx) {
  int32_t code = 0;
  
D
dapan1121 已提交
1044
  if (node->pLeft && SCL_IS_VAR_VALUE_NODE(node->pLeft)) {
1045 1046 1047
    if (node->pRight && (TSDB_DATA_TYPE_TIMESTAMP == ((SExprNode *)node->pRight)->resType.type)) {
      SCL_ERR_JRET(sclConvertToTsValueNode(((SExprNode *)node->pRight)->resType.precision, (SValueNode*)node->pLeft));
    }
D
dapan1121 已提交
1048
  } else if (node->pRight && SCL_IS_NOTNULL_CONST_NODE(node->pRight)) {
1049 1050 1051
    if (node->pLeft && (TSDB_DATA_TYPE_TIMESTAMP == ((SExprNode *)node->pLeft)->resType.type)) {
      if (SCL_IS_VAR_VALUE_NODE(node->pRight)) {
        SCL_ERR_JRET(sclConvertToTsValueNode(((SExprNode *)node->pLeft)->resType.precision, (SValueNode*)node->pRight));
D
dapan1121 已提交
1052
      } else if (QUERY_NODE_NODE_LIST == node->pRight->type) {
1053 1054 1055 1056 1057 1058 1059 1060 1061 1062 1063 1064 1065 1066 1067 1068 1069 1070 1071 1072 1073 1074 1075 1076 1077 1078 1079 1080 1081 1082 1083 1084 1085 1086 1087 1088 1089 1090 1091 1092 1093 1094 1095 1096 1097 1098 1099 1100 1101 1102 1103 1104 1105
        SNode* pNode;
        FOREACH(pNode, ((SNodeListNode*)node->pRight)->pNodeList) {
          if (SCL_IS_VAR_VALUE_NODE(pNode)) {
            SCL_ERR_JRET(sclConvertToTsValueNode(((SExprNode *)node->pLeft)->resType.precision, (SValueNode*)pNode));
          }
        }
      }
    }
  }

  return TSDB_CODE_SUCCESS;

_return:

  ctx->code = code;
  return DEAL_RES_ERROR;
}



int32_t sclConvertCaseWhenValueNodeTs(SCaseWhenNode *node, SScalarCtx *ctx) {
  int32_t code = 0;

  if (NULL == node->pCase) {
    return TSDB_CODE_SUCCESS;
  }
  
  if (SCL_IS_VAR_VALUE_NODE(node->pCase)) {
    SNode* pNode;
    FOREACH(pNode, node->pWhenThenList) {
      SExprNode *pExpr = (SExprNode *)((SWhenThenNode *)pNode)->pWhen;
      if (TSDB_DATA_TYPE_TIMESTAMP == pExpr->resType.type) {
        SCL_ERR_JRET(sclConvertToTsValueNode(pExpr->resType.precision, (SValueNode*)node->pCase));
        break;
      }
    }
  } else if (TSDB_DATA_TYPE_TIMESTAMP == ((SExprNode *)node->pCase)->resType.type) {
    SNode* pNode;
    FOREACH(pNode, node->pWhenThenList) {
      if (SCL_IS_VAR_VALUE_NODE(((SWhenThenNode *)pNode)->pWhen)) {
        SCL_ERR_JRET(sclConvertToTsValueNode(((SExprNode *)node->pCase)->resType.precision, (SValueNode*)((SWhenThenNode *)pNode)->pWhen));
      }
    }
  }

  return TSDB_CODE_SUCCESS;

_return:

  ctx->code = code;
  return DEAL_RES_ERROR;
}

H
Hongze Cheng 已提交
1106
EDealRes sclRewriteNonConstOperator(SNode **pNode, SScalarCtx *ctx) {
D
dapan1121 已提交
1107
  SOperatorNode *node = (SOperatorNode *)*pNode;
H
Hongze Cheng 已提交
1108
  int32_t        code = 0;
D
dapan1121 已提交
1109 1110 1111

  if (node->pLeft && (QUERY_NODE_VALUE == nodeType(node->pLeft))) {
    SValueNode *valueNode = (SValueNode *)node->pLeft;
H
Hongze Cheng 已提交
1112 1113
    if (SCL_IS_NULL_VALUE_NODE(valueNode) && (node->opType != OP_TYPE_IS_NULL && node->opType != OP_TYPE_IS_NOT_NULL) &&
        (!sclContainsAggFuncNode(node->pRight))) {
D
dapan1121 已提交
1114
      return sclRewriteNullInOptr(pNode, ctx, node->opType);
D
dapan1121 已提交
1115
    }
D
dapan 已提交
1116

1117 1118
    if (SCL_IS_COMPARISON_OPERATOR(node->opType) && SCL_DOWNGRADE_DATETYPE(valueNode->node.resType.type)) {
      sclDowngradeValueType(valueNode);
H
Hongze Cheng 已提交
1119
    }
D
dapan1121 已提交
1120 1121 1122 1123
  }

  if (node->pRight && (QUERY_NODE_VALUE == nodeType(node->pRight))) {
    SValueNode *valueNode = (SValueNode *)node->pRight;
H
Hongze Cheng 已提交
1124 1125
    if (SCL_IS_NULL_VALUE_NODE(valueNode) && (node->opType != OP_TYPE_IS_NULL && node->opType != OP_TYPE_IS_NOT_NULL) &&
        (!sclContainsAggFuncNode(node->pLeft))) {
D
dapan1121 已提交
1126
      return sclRewriteNullInOptr(pNode, ctx, node->opType);
D
dapan1121 已提交
1127
    }
D
dapan 已提交
1128

1129 1130
    if (SCL_IS_COMPARISON_OPERATOR(node->opType) && SCL_DOWNGRADE_DATETYPE(valueNode->node.resType.type)) {
      sclDowngradeValueType(valueNode);
H
Hongze Cheng 已提交
1131
    }
D
dapan1121 已提交
1132 1133 1134 1135
  }

  if (node->pRight && (QUERY_NODE_NODE_LIST == nodeType(node->pRight))) {
    SNodeListNode *listNode = (SNodeListNode *)node->pRight;
H
Hongze Cheng 已提交
1136
    SNode         *tnode = NULL;
D
dapan1121 已提交
1137 1138 1139 1140 1141
    WHERE_EACH(tnode, listNode->pNodeList) {
      if (SCL_IS_NULL_VALUE_NODE(tnode)) {
        if (node->opType == OP_TYPE_IN) {
          ERASE_NODE(listNode->pNodeList);
          continue;
H
Hongze Cheng 已提交
1142
        } else {  // OP_TYPE_NOT_IN
D
dapan1121 已提交
1143
          return sclRewriteNullInOptr(pNode, ctx, node->opType);
D
dapan1121 已提交
1144 1145 1146 1147 1148 1149 1150
        }
      }

      WHERE_NEXT;
    }

    if (listNode->pNodeList->length <= 0) {
D
dapan1121 已提交
1151
      return sclRewriteNullInOptr(pNode, ctx, node->opType);
D
dapan1121 已提交
1152 1153 1154 1155 1156 1157
    }
  }

  return DEAL_RES_CONTINUE;
}

H
Hongze Cheng 已提交
1158
EDealRes sclRewriteFunction(SNode **pNode, SScalarCtx *ctx) {
D
dapan1121 已提交
1159
  SFunctionNode *node = (SFunctionNode *)*pNode;
H
Hongze Cheng 已提交
1160
  SNode         *tnode = NULL;
X
Xiaoyu Wang 已提交
1161
  if ((!fmIsScalarFunc(node->funcId) && (!ctx->dual)) || fmIsUserDefinedFunc(node->funcId)) {
G
Ganlin Zhao 已提交
1162 1163
    return DEAL_RES_CONTINUE;
  }
1164

D
dapan1121 已提交
1165 1166 1167 1168 1169 1170
  FOREACH(tnode, node->pParameterList) {
    if (!SCL_IS_CONST_NODE(tnode)) {
      return DEAL_RES_CONTINUE;
    }
  }

D
dapan1121 已提交
1171
  SScalarParam output = {0};
1172

1173
  ctx->code = sclExecFunction(node, ctx, &output);
D
dapan 已提交
1174
  if (ctx->code) {
D
dapan1121 已提交
1175 1176 1177
    return DEAL_RES_ERROR;
  }

D
dapan1121 已提交
1178
  SValueNode *res = (SValueNode *)nodesMakeNode(QUERY_NODE_VALUE);
D
dapan1121 已提交
1179 1180
  if (NULL == res) {
    sclError("make value node failed");
D
dapan1121 已提交
1181
    sclFreeParam(&output);
S
Shengliang Guan 已提交
1182
    ctx->code = TSDB_CODE_OUT_OF_MEMORY;
D
dapan1121 已提交
1183 1184 1185
    return DEAL_RES_ERROR;
  }

1186 1187
  res->translate = true;

1188 1189 1190 1191
  res->node.resType.type = output.columnData->info.type;
  res->node.resType.bytes = output.columnData->info.bytes;
  res->node.resType.scale = output.columnData->info.scale;
  res->node.resType.precision = output.columnData->info.precision;
1192
  if (colDataIsNull_s(output.columnData, 0)) {
1193
    res->isNull = true;
D
dapan1121 已提交
1194
  } else {
1195
    int32_t type = output.columnData->info.type;
H
Hongze Cheng 已提交
1196
    if (type == TSDB_DATA_TYPE_JSON) {
wmmhello's avatar
wmmhello 已提交
1197 1198 1199 1200
      int32_t len = getJsonValueLen(output.columnData->pData);
      res->datum.p = taosMemoryCalloc(len, 1);
      memcpy(res->datum.p, output.columnData->pData, len);
    } else if (IS_VAR_DATA_TYPE(type)) {
H
Hongze Cheng 已提交
1201
      // res->datum.p = taosMemoryCalloc(res->node.resType.bytes + VARSTR_HEADER_SIZE + 1, 1);
1202
      res->datum.p = taosMemoryCalloc(varDataTLen(output.columnData->pData) + 1, 1);
1203
      res->node.resType.bytes = varDataTLen(output.columnData->pData);
1204
      memcpy(res->datum.p, output.columnData->pData, varDataTLen(output.columnData->pData));
1205
    } else {
D
dapan1121 已提交
1206
      nodesSetValueNodeValue(res, output.columnData->pData);
1207
    }
D
dapan1121 已提交
1208
  }
1209

D
dapan1121 已提交
1210
  nodesDestroyNode(*pNode);
H
Hongze Cheng 已提交
1211
  *pNode = (SNode *)res;
D
dapan1121 已提交
1212

D
dapan1121 已提交
1213
  sclFreeParam(&output);
D
dapan1121 已提交
1214 1215 1216
  return DEAL_RES_CONTINUE;
}

H
Hongze Cheng 已提交
1217
EDealRes sclRewriteLogic(SNode **pNode, SScalarCtx *ctx) {
D
dapan1121 已提交
1218 1219
  SLogicConditionNode *node = (SLogicConditionNode *)*pNode;

H
Haojun Liao 已提交
1220
  SScalarParam output = {0};
D
dapan 已提交
1221 1222
  ctx->code = sclExecLogic(node, ctx, &output);
  if (ctx->code) {
D
dapan1121 已提交
1223 1224 1225
    return DEAL_RES_ERROR;
  }

D
dapan1121 已提交
1226 1227 1228 1229
  if (0 == output.numOfRows) {
    return DEAL_RES_CONTINUE;
  }

D
dapan1121 已提交
1230
  SValueNode *res = (SValueNode *)nodesMakeNode(QUERY_NODE_VALUE);
D
dapan1121 已提交
1231 1232
  if (NULL == res) {
    sclError("make value node failed");
1233
    sclFreeParam(&output);
S
Shengliang Guan 已提交
1234
    ctx->code = TSDB_CODE_OUT_OF_MEMORY;
D
dapan1121 已提交
1235 1236 1237 1238
    return DEAL_RES_ERROR;
  }

  res->node.resType = node->node.resType;
1239
  res->translate = true;
D
dapan1121 已提交
1240

1241 1242 1243 1244
  int32_t type = output.columnData->info.type;
  if (IS_VAR_DATA_TYPE(type)) {
    res->datum.p = output.columnData->pData;
    output.columnData->pData = NULL;
D
dapan1121 已提交
1245
  } else {
D
dapan1121 已提交
1246
    nodesSetValueNodeValue(res, output.columnData->pData);
D
dapan1121 已提交
1247
  }
D
dapan1121 已提交
1248 1249

  nodesDestroyNode(*pNode);
H
Hongze Cheng 已提交
1250
  *pNode = (SNode *)res;
D
dapan1121 已提交
1251

D
dapan1121 已提交
1252
  sclFreeParam(&output);
D
dapan1121 已提交
1253 1254 1255
  return DEAL_RES_CONTINUE;
}

1256

H
Hongze Cheng 已提交
1257
EDealRes sclRewriteOperator(SNode **pNode, SScalarCtx *ctx) {
D
dapan1121 已提交
1258
  SOperatorNode *node = (SOperatorNode *)*pNode;
D
dapan1121 已提交
1259

1260 1261
  SCL_ERR_RET(sclConvertOpValueNodeTs(node, ctx));

D
dapan1121 已提交
1262
  if ((!SCL_IS_CONST_NODE(node->pLeft)) || (!SCL_IS_CONST_NODE(node->pRight))) {
D
dapan 已提交
1263
    return sclRewriteNonConstOperator(pNode, ctx);
D
dapan1121 已提交
1264 1265
  }

H
Haojun Liao 已提交
1266
  SScalarParam output = {0};
D
dapan 已提交
1267 1268
  ctx->code = sclExecOperator(node, ctx, &output);
  if (ctx->code) {
dengyihao's avatar
dengyihao 已提交
1269
    sclFreeParam(&output);
D
dapan1121 已提交
1270 1271 1272
    return DEAL_RES_ERROR;
  }

D
dapan1121 已提交
1273
  SValueNode *res = (SValueNode *)nodesMakeNode(QUERY_NODE_VALUE);
D
dapan1121 已提交
1274
  if (NULL == res) {
1275 1276
    sclError("make value node failed");
    sclFreeParam(&output);
S
Shengliang Guan 已提交
1277
    ctx->code = TSDB_CODE_OUT_OF_MEMORY;
D
dapan1121 已提交
1278 1279 1280
    return DEAL_RES_ERROR;
  }

1281
  res->translate = true;
D
dapan1121 已提交
1282

1283
  res->node.resType = node->node.resType;
1284
  if (colDataIsNull_s(output.columnData, 0)) {
1285 1286
    res->isNull = true;
    res->node.resType = node->node.resType;
1287
  } else {
1288 1289 1290 1291 1292
    int32_t type = output.columnData->info.type;
    if (IS_VAR_DATA_TYPE(type)) {  // todo refactor
      res->datum.p = output.columnData->pData;
      output.columnData->pData = NULL;
    } else {
G
Ganlin Zhao 已提交
1293
      nodesSetValueNodeValue(res, output.columnData->pData);
1294
    }
D
dapan1121 已提交
1295
  }
D
dapan1121 已提交
1296 1297

  nodesDestroyNode(*pNode);
H
Hongze Cheng 已提交
1298
  *pNode = (SNode *)res;
D
dapan1121 已提交
1299

H
Haojun Liao 已提交
1300
  sclFreeParam(&output);
D
dapan1121 已提交
1301 1302 1303
  return DEAL_RES_CONTINUE;
}

X
Xiaoyu Wang 已提交
1304
EDealRes sclRewriteCaseWhen(SNode **pNode, SScalarCtx *ctx) {
D
dapan1121 已提交
1305 1306
  SCaseWhenNode *node = (SCaseWhenNode *)*pNode;

1307 1308
  SCL_ERR_RET(sclConvertCaseWhenValueNodeTs(node, ctx));

D
dapan1121 已提交
1309 1310 1311 1312
  if ((!SCL_IS_CONST_NODE(node->pCase)) || (!SCL_IS_CONST_NODE(node->pElse))) {
    return DEAL_RES_CONTINUE;
  }

X
Xiaoyu Wang 已提交
1313
  SNode *tnode = NULL;
D
dapan1121 已提交
1314
  FOREACH(tnode, node->pWhenThenList) {
X
Xiaoyu Wang 已提交
1315
    SWhenThenNode *pWhenThen = (SWhenThenNode *)tnode;
D
dapan1121 已提交
1316
    if (!SCL_IS_CONST_NODE(pWhenThen->pWhen) || !SCL_IS_CONST_NODE(pWhenThen->pThen)) {
D
dapan1121 已提交
1317 1318
      return DEAL_RES_CONTINUE;
    }
X
Xiaoyu Wang 已提交
1319
  }
D
dapan1121 已提交
1320 1321 1322 1323 1324 1325 1326 1327 1328 1329 1330

  SScalarParam output = {0};
  ctx->code = sclExecCaseWhen(node, ctx, &output);
  if (ctx->code) {
    return DEAL_RES_ERROR;
  }

  SValueNode *res = (SValueNode *)nodesMakeNode(QUERY_NODE_VALUE);
  if (NULL == res) {
    sclError("make value node failed");
    sclFreeParam(&output);
S
Shengliang Guan 已提交
1331
    ctx->code = TSDB_CODE_OUT_OF_MEMORY;
D
dapan1121 已提交
1332 1333 1334 1335 1336 1337 1338 1339 1340 1341 1342 1343 1344 1345 1346 1347 1348 1349 1350 1351
    return DEAL_RES_ERROR;
  }

  res->translate = true;

  res->node.resType = node->node.resType;
  if (colDataIsNull_s(output.columnData, 0)) {
    res->isNull = true;
    res->node.resType = node->node.resType;
  } else {
    int32_t type = output.columnData->info.type;
    if (IS_VAR_DATA_TYPE(type)) {  // todo refactor
      res->datum.p = output.columnData->pData;
      output.columnData->pData = NULL;
    } else {
      nodesSetValueNodeValue(res, output.columnData->pData);
    }
  }

  nodesDestroyNode(*pNode);
X
Xiaoyu Wang 已提交
1352
  *pNode = (SNode *)res;
D
dapan1121 已提交
1353 1354 1355 1356 1357

  sclFreeParam(&output);
  return DEAL_RES_CONTINUE;
}

H
Hongze Cheng 已提交
1358
EDealRes sclConstantsRewriter(SNode **pNode, void *pContext) {
D
dapan 已提交
1359 1360
  SScalarCtx *ctx = (SScalarCtx *)pContext;

D
dapan1121 已提交
1361 1362 1363 1364
  if (QUERY_NODE_OPERATOR == nodeType(*pNode)) {
    return sclRewriteOperator(pNode, ctx);
  }

D
dapan1121 已提交
1365
  if (QUERY_NODE_FUNCTION == nodeType(*pNode)) {
D
dapan 已提交
1366
    return sclRewriteFunction(pNode, ctx);
D
dapan1121 已提交
1367 1368 1369
  }

  if (QUERY_NODE_LOGIC_CONDITION == nodeType(*pNode)) {
D
dapan 已提交
1370
    return sclRewriteLogic(pNode, ctx);
D
dapan1121 已提交
1371 1372
  }

D
dapan1121 已提交
1373
  if (QUERY_NODE_CASE_WHEN == nodeType(*pNode)) {
D
dapan1121 已提交
1374
    return sclRewriteCaseWhen(pNode, ctx);
1375
  }
1376 1377

  return DEAL_RES_CONTINUE;
D
dapan1121 已提交
1378 1379
}

H
Hongze Cheng 已提交
1380
EDealRes sclWalkFunction(SNode *pNode, SScalarCtx *ctx) {
D
dapan1121 已提交
1381
  SFunctionNode *node = (SFunctionNode *)pNode;
H
Hongze Cheng 已提交
1382
  SScalarParam   output = {0};
1383

1384
  ctx->code = sclExecFunction(node, ctx, &output);
D
dapan1121 已提交
1385 1386
  if (ctx->code) {
    return DEAL_RES_ERROR;
D
dapan1121 已提交
1387 1388
  }

D
dapan1121 已提交
1389
  if (taosHashPut(ctx->pRes, &pNode, POINTER_BYTES, &output, sizeof(output))) {
S
Shengliang Guan 已提交
1390
    ctx->code = TSDB_CODE_OUT_OF_MEMORY;
D
dapan1121 已提交
1391
    return DEAL_RES_ERROR;
D
dapan1121 已提交
1392 1393
  }

D
dapan1121 已提交
1394 1395 1396
  return DEAL_RES_CONTINUE;
}

H
Hongze Cheng 已提交
1397
EDealRes sclWalkLogic(SNode *pNode, SScalarCtx *ctx) {
D
dapan1121 已提交
1398
  SLogicConditionNode *node = (SLogicConditionNode *)pNode;
H
Hongze Cheng 已提交
1399
  SScalarParam         output = {0};
1400

D
dapan1121 已提交
1401 1402 1403
  ctx->code = sclExecLogic(node, ctx, &output);
  if (ctx->code) {
    return DEAL_RES_ERROR;
D
dapan1121 已提交
1404 1405
  }

D
dapan1121 已提交
1406
  if (taosHashPut(ctx->pRes, &pNode, POINTER_BYTES, &output, sizeof(output))) {
S
Shengliang Guan 已提交
1407
    ctx->code = TSDB_CODE_OUT_OF_MEMORY;
D
dapan1121 已提交
1408
    return DEAL_RES_ERROR;
D
dapan1121 已提交
1409 1410 1411 1412 1413
  }

  return DEAL_RES_CONTINUE;
}

H
Hongze Cheng 已提交
1414
EDealRes sclWalkOperator(SNode *pNode, SScalarCtx *ctx) {
D
dapan1121 已提交
1415
  SOperatorNode *node = (SOperatorNode *)pNode;
H
Hongze Cheng 已提交
1416
  SScalarParam   output = {0};
G
Ganlin Zhao 已提交
1417

D
dapan1121 已提交
1418 1419
  ctx->code = sclExecOperator(node, ctx, &output);
  if (ctx->code) {
dengyihao's avatar
dengyihao 已提交
1420
    sclFreeParam(&output);
D
dapan1121 已提交
1421 1422
    return DEAL_RES_ERROR;
  }
D
dapan1121 已提交
1423

D
dapan1121 已提交
1424
  if (taosHashPut(ctx->pRes, &pNode, POINTER_BYTES, &output, sizeof(output))) {
S
Shengliang Guan 已提交
1425
    ctx->code = TSDB_CODE_OUT_OF_MEMORY;
D
dapan1121 已提交
1426 1427 1428
    return DEAL_RES_ERROR;
  }

D
dapan1121 已提交
1429 1430 1431
  return DEAL_RES_CONTINUE;
}

H
Hongze Cheng 已提交
1432
EDealRes sclWalkTarget(SNode *pNode, SScalarCtx *ctx) {
1433 1434 1435
  STargetNode *target = (STargetNode *)pNode;

  if (target->dataBlockId >= taosArrayGetSize(ctx->pBlockList)) {
H
Hongze Cheng 已提交
1436 1437
    sclError("target tupleId is too big, tupleId:%d, dataBlockNum:%d", target->dataBlockId,
             (int32_t)taosArrayGetSize(ctx->pBlockList));
1438 1439 1440 1441 1442
    ctx->code = TSDB_CODE_QRY_INVALID_INPUT;
    return DEAL_RES_ERROR;
  }

  int32_t index = -1;
H
Hongze Cheng 已提交
1443 1444
  for (int32_t i = 0; i < taosArrayGetSize(ctx->pBlockList); ++i) {
    SSDataBlock *pb = taosArrayGetP(ctx->pBlockList, i);
H
Haojun Liao 已提交
1445
    if (pb->info.id.blockId == target->dataBlockId) {
1446 1447 1448 1449 1450 1451
      index = i;
      break;
    }
  }

  if (index == -1) {
H
Hongze Cheng 已提交
1452 1453
    sclError("column tupleId is too big, tupleId:%d, dataBlockNum:%d", target->dataBlockId,
             (int32_t)taosArrayGetSize(ctx->pBlockList));
1454 1455 1456 1457 1458 1459 1460
    ctx->code = TSDB_CODE_QRY_INVALID_INPUT;
    return DEAL_RES_ERROR;
  }

  SSDataBlock *block = *(SSDataBlock **)taosArrayGet(ctx->pBlockList, index);

  if (target->slotId >= taosArrayGetSize(block->pDataBlock)) {
H
Hongze Cheng 已提交
1461 1462
    sclError("target slot not exist, dataBlockId:%d, slotId:%d, dataBlockNum:%d", target->dataBlockId, target->slotId,
             (int32_t)taosArrayGetSize(block->pDataBlock));
1463 1464 1465 1466 1467 1468 1469 1470 1471 1472
    ctx->code = TSDB_CODE_QRY_INVALID_INPUT;
    return DEAL_RES_ERROR;
  }

  // if block->pDataBlock is not enough, there are problems if target->slotId bigger than the size of block->pDataBlock,
  SColumnInfoData *col = taosArrayGet(block->pDataBlock, target->slotId);

  SScalarParam *res = (SScalarParam *)taosHashGet(ctx->pRes, (void *)&target->pExpr, POINTER_BYTES);
  if (NULL == res) {
    sclError("no valid res in hash, node:%p, type:%d", target->pExpr, nodeType(target->pExpr));
S
Shengliang Guan 已提交
1473
    ctx->code = TSDB_CODE_APP_ERROR;
1474 1475 1476
    return DEAL_RES_ERROR;
  }

1477
  colDataAssign(col, res->columnData, res->numOfRows, NULL);
1478 1479 1480 1481 1482 1483
  block->info.rows = res->numOfRows;

  sclFreeParam(res);
  taosHashRemove(ctx->pRes, (void *)&target->pExpr, POINTER_BYTES);
  return DEAL_RES_CONTINUE;
}
D
dapan 已提交
1484

X
Xiaoyu Wang 已提交
1485
EDealRes sclWalkCaseWhen(SNode *pNode, SScalarCtx *ctx) {
D
dapan1121 已提交
1486
  SCaseWhenNode *node = (SCaseWhenNode *)pNode;
X
Xiaoyu Wang 已提交
1487
  SScalarParam   output = {0};
D
dapan1121 已提交
1488 1489 1490 1491 1492 1493 1494

  ctx->code = sclExecCaseWhen(node, ctx, &output);
  if (ctx->code) {
    return DEAL_RES_ERROR;
  }

  if (taosHashPut(ctx->pRes, &pNode, POINTER_BYTES, &output, sizeof(output))) {
S
Shengliang Guan 已提交
1495
    ctx->code = TSDB_CODE_OUT_OF_MEMORY;
D
dapan1121 已提交
1496 1497 1498 1499 1500 1501
    return DEAL_RES_ERROR;
  }

  return DEAL_RES_CONTINUE;
}

H
Hongze Cheng 已提交
1502
EDealRes sclCalcWalker(SNode *pNode, void *pContext) {
X
Xiaoyu Wang 已提交
1503 1504 1505
  if (QUERY_NODE_VALUE == nodeType(pNode) || QUERY_NODE_NODE_LIST == nodeType(pNode) ||
      QUERY_NODE_COLUMN == nodeType(pNode) || QUERY_NODE_LEFT_VALUE == nodeType(pNode) ||
      QUERY_NODE_WHEN_THEN == nodeType(pNode)) {
D
dapan1121 已提交
1506
    return DEAL_RES_CONTINUE;
D
dapan1121 已提交
1507
  }
D
dapan 已提交
1508 1509

  SScalarCtx *ctx = (SScalarCtx *)pContext;
D
dapan1121 已提交
1510 1511 1512 1513
  if (QUERY_NODE_OPERATOR == nodeType(pNode)) {
    return sclWalkOperator(pNode, ctx);
  }

D
dapan1121 已提交
1514
  if (QUERY_NODE_FUNCTION == nodeType(pNode)) {
D
dapan 已提交
1515
    return sclWalkFunction(pNode, ctx);
D
dapan1121 已提交
1516
  }
D
dapan1121 已提交
1517

D
dapan1121 已提交
1518
  if (QUERY_NODE_LOGIC_CONDITION == nodeType(pNode)) {
D
dapan 已提交
1519
    return sclWalkLogic(pNode, ctx);
D
dapan1121 已提交
1520
  }
D
dapan1121 已提交
1521

1522 1523 1524
  if (QUERY_NODE_TARGET == nodeType(pNode)) {
    return sclWalkTarget(pNode, ctx);
  }
D
dapan1121 已提交
1525

D
dapan1121 已提交
1526 1527 1528 1529
  if (QUERY_NODE_CASE_WHEN == nodeType(pNode)) {
    return sclWalkCaseWhen(pNode, ctx);
  }

D
dapan 已提交
1530
  sclError("invalid node type for scalar calculating, type:%d", nodeType(pNode));
D
dapan1121 已提交
1531 1532
  ctx->code = TSDB_CODE_QRY_INVALID_INPUT;
  return DEAL_RES_ERROR;
D
dapan1121 已提交
1533 1534
}

D
dapan1121 已提交
1535
int32_t sclCalcConstants(SNode *pNode, bool dual, SNode **pRes) {
D
dapan1121 已提交
1536 1537 1538 1539
  if (NULL == pNode) {
    SCL_ERR_RET(TSDB_CODE_QRY_INVALID_INPUT);
  }

H
Hongze Cheng 已提交
1540
  int32_t    code = 0;
D
dapan 已提交
1541
  SScalarCtx ctx = {0};
D
dapan1121 已提交
1542
  ctx.dual = dual;
1543
  ctx.pRes = taosHashInit(SCL_DEFAULT_OP_NUM, taosGetDefaultHashFunction(TSDB_DATA_TYPE_BIGINT), false, HASH_NO_LOCK);
D
dapan 已提交
1544
  if (NULL == ctx.pRes) {
1545
    sclError("taosHashInit failed, num:%d", SCL_DEFAULT_OP_NUM);
S
Shengliang Guan 已提交
1546
    SCL_ERR_RET(TSDB_CODE_OUT_OF_MEMORY);
D
dapan 已提交
1547
  }
G
Ganlin Zhao 已提交
1548

X
Xiaoyu Wang 已提交
1549
  nodesRewriteExprPostOrder(&pNode, sclConstantsRewriter, (void *)&ctx);
D
dapan 已提交
1550
  SCL_ERR_JRET(ctx.code);
D
dapan1121 已提交
1551 1552
  *pRes = pNode;

D
dapan 已提交
1553
_return:
1554

D
dapan 已提交
1555 1556
  sclFreeRes(ctx.pRes);
  return code;
D
dapan1121 已提交
1557
}
D
dapan1121 已提交
1558

H
Hongze Cheng 已提交
1559 1560
static int32_t sclGetMinusOperatorResType(SOperatorNode *pOp) {
  if (!IS_MATHABLE_TYPE(((SExprNode *)(pOp->pLeft))->resType.type)) {
1561 1562 1563 1564 1565 1566 1567
    return TSDB_CODE_TSC_INVALID_OPERATION;
  }
  pOp->node.resType.type = TSDB_DATA_TYPE_DOUBLE;
  pOp->node.resType.bytes = tDataTypes[TSDB_DATA_TYPE_DOUBLE].bytes;
  return TSDB_CODE_SUCCESS;
}

H
Hongze Cheng 已提交
1568
static int32_t sclGetMathOperatorResType(SOperatorNode *pOp) {
G
Ganlin Zhao 已提交
1569 1570 1571 1572
  if (pOp == NULL || pOp->pLeft == NULL || pOp->pRight == NULL) {
    return TSDB_CODE_TSC_INVALID_OPERATION;
  }

H
Hongze Cheng 已提交
1573 1574
  SDataType ldt = ((SExprNode *)(pOp->pLeft))->resType;
  SDataType rdt = ((SExprNode *)(pOp->pRight))->resType;
G
Ganlin Zhao 已提交
1575

1576 1577 1578 1579 1580 1581 1582 1583 1584 1585 1586 1587 1588 1589 1590 1591 1592 1593 1594
  if ((TSDB_DATA_TYPE_TIMESTAMP == ldt.type && TSDB_DATA_TYPE_TIMESTAMP == rdt.type) ||
      (TSDB_DATA_TYPE_TIMESTAMP == ldt.type && (IS_VAR_DATA_TYPE(rdt.type) || IS_FLOAT_TYPE(rdt.type))) ||
      (TSDB_DATA_TYPE_TIMESTAMP == rdt.type && (IS_VAR_DATA_TYPE(ldt.type) || IS_FLOAT_TYPE(ldt.type)))) {
    return TSDB_CODE_TSC_INVALID_OPERATION;
  }

  if ((TSDB_DATA_TYPE_TIMESTAMP == ldt.type && IS_INTEGER_TYPE(rdt.type)) ||
      (TSDB_DATA_TYPE_TIMESTAMP == rdt.type && IS_INTEGER_TYPE(ldt.type)) ||
      (TSDB_DATA_TYPE_TIMESTAMP == ldt.type && TSDB_DATA_TYPE_BOOL == rdt.type) ||
      (TSDB_DATA_TYPE_TIMESTAMP == rdt.type && TSDB_DATA_TYPE_BOOL == ldt.type)) {
    pOp->node.resType.type = TSDB_DATA_TYPE_TIMESTAMP;
    pOp->node.resType.bytes = tDataTypes[TSDB_DATA_TYPE_TIMESTAMP].bytes;
  } else {
    pOp->node.resType.type = TSDB_DATA_TYPE_DOUBLE;
    pOp->node.resType.bytes = tDataTypes[TSDB_DATA_TYPE_DOUBLE].bytes;
  }
  return TSDB_CODE_SUCCESS;
}

H
Hongze Cheng 已提交
1595
static int32_t sclGetCompOperatorResType(SOperatorNode *pOp) {
G
Ganlin Zhao 已提交
1596
  if (pOp == NULL || pOp->pLeft == NULL) {
G
Ganlin Zhao 已提交
1597 1598 1599
    return TSDB_CODE_TSC_INVALID_OPERATION;
  }

H
Hongze Cheng 已提交
1600
  SDataType ldt = ((SExprNode *)(pOp->pLeft))->resType;
G
Ganlin Zhao 已提交
1601

1602
  if (OP_TYPE_IN == pOp->opType || OP_TYPE_NOT_IN == pOp->opType) {
G
Ganlin Zhao 已提交
1603 1604 1605
    if (pOp->pRight == NULL) {
      return TSDB_CODE_TSC_INVALID_OPERATION;
    }
H
Hongze Cheng 已提交
1606
    ((SExprNode *)(pOp->pRight))->resType = ldt;
1607
  } else if (nodesIsRegularOp(pOp)) {
G
Ganlin Zhao 已提交
1608 1609 1610
    if (pOp->pRight == NULL) {
      return TSDB_CODE_TSC_INVALID_OPERATION;
    }
H
Hongze Cheng 已提交
1611
    SDataType rdt = ((SExprNode *)(pOp->pRight))->resType;
1612 1613 1614
    if (!IS_VAR_DATA_TYPE(ldt.type) || QUERY_NODE_VALUE != nodeType(pOp->pRight) ||
        (!IS_STR_DATA_TYPE(rdt.type) && (rdt.type != TSDB_DATA_TYPE_NULL))) {
      return TSDB_CODE_TSC_INVALID_OPERATION;
1615 1616 1617 1618 1619 1620 1621
    }
  }
  pOp->node.resType.type = TSDB_DATA_TYPE_BOOL;
  pOp->node.resType.bytes = tDataTypes[TSDB_DATA_TYPE_BOOL].bytes;
  return TSDB_CODE_SUCCESS;
}

H
Hongze Cheng 已提交
1622
static int32_t sclGetJsonOperatorResType(SOperatorNode *pOp) {
G
Ganlin Zhao 已提交
1623 1624 1625 1626
  if (pOp == NULL || pOp->pLeft == NULL || pOp->pRight == NULL) {
    return TSDB_CODE_TSC_INVALID_OPERATION;
  }

H
Hongze Cheng 已提交
1627 1628
  SDataType ldt = ((SExprNode *)(pOp->pLeft))->resType;
  SDataType rdt = ((SExprNode *)(pOp->pRight))->resType;
G
Ganlin Zhao 已提交
1629

1630 1631 1632 1633 1634 1635 1636 1637 1638 1639 1640 1641
  if (TSDB_DATA_TYPE_JSON != ldt.type || !IS_STR_DATA_TYPE(rdt.type)) {
    return TSDB_CODE_TSC_INVALID_OPERATION;
  }
  if (pOp->opType == OP_TYPE_JSON_GET_VALUE) {
    pOp->node.resType.type = TSDB_DATA_TYPE_JSON;
  } else if (pOp->opType == OP_TYPE_JSON_CONTAINS) {
    pOp->node.resType.type = TSDB_DATA_TYPE_BOOL;
  }
  pOp->node.resType.bytes = tDataTypes[pOp->node.resType.type].bytes;
  return TSDB_CODE_SUCCESS;
}

H
Hongze Cheng 已提交
1642
static int32_t sclGetBitwiseOperatorResType(SOperatorNode *pOp) {
1643 1644 1645 1646 1647
  pOp->node.resType.type = TSDB_DATA_TYPE_BIGINT;
  pOp->node.resType.bytes = tDataTypes[TSDB_DATA_TYPE_BIGINT].bytes;
  return TSDB_CODE_SUCCESS;
}

H
Hongze Cheng 已提交
1648
int32_t scalarCalculateConstants(SNode *pNode, SNode **pRes) { return sclCalcConstants(pNode, false, pRes); }
D
dapan1121 已提交
1649

H
Hongze Cheng 已提交
1650
int32_t scalarCalculateConstantsFromDual(SNode *pNode, SNode **pRes) { return sclCalcConstants(pNode, true, pRes); }
D
dapan1121 已提交
1651 1652 1653 1654 1655 1656

int32_t scalarCalculate(SNode *pNode, SArray *pBlockList, SScalarParam *pDst) {
  if (NULL == pNode || NULL == pBlockList) {
    SCL_ERR_RET(TSDB_CODE_QRY_INVALID_INPUT);
  }

H
Hongze Cheng 已提交
1657
  int32_t    code = 0;
D
dapan1121 已提交
1658 1659 1660 1661 1662 1663
  SScalarCtx ctx = {.code = 0, .pBlockList = pBlockList, .param = pDst ? pDst->param : NULL};

  // TODO: OPT performance
  ctx.pRes = taosHashInit(SCL_DEFAULT_OP_NUM, taosGetDefaultHashFunction(TSDB_DATA_TYPE_BIGINT), false, HASH_NO_LOCK);
  if (NULL == ctx.pRes) {
    sclError("taosHashInit failed, num:%d", SCL_DEFAULT_OP_NUM);
S
Shengliang Guan 已提交
1664
    SCL_ERR_RET(TSDB_CODE_OUT_OF_MEMORY);
D
dapan1121 已提交
1665
  }
1666

D
dapan1121 已提交
1667 1668 1669 1670 1671 1672 1673
  nodesWalkExprPostOrder(pNode, sclCalcWalker, (void *)&ctx);
  SCL_ERR_JRET(ctx.code);

  if (pDst) {
    SScalarParam *res = (SScalarParam *)taosHashGet(ctx.pRes, (void *)&pNode, POINTER_BYTES);
    if (NULL == res) {
      sclError("no valid res in hash, node:%p, type:%d", pNode, nodeType(pNode));
S
Shengliang Guan 已提交
1674
      SCL_ERR_JRET(TSDB_CODE_APP_ERROR);
D
dapan1121 已提交
1675 1676 1677 1678 1679
    }

    if (1 == res->numOfRows) {
      SCL_ERR_JRET(sclExtendResRows(pDst, res, pBlockList));
    } else {
H
Haojun Liao 已提交
1680
      colInfoDataEnsureCapacity(pDst->columnData, res->numOfRows, true);
D
dapan1121 已提交
1681 1682
      colDataAssign(pDst->columnData, res->columnData, res->numOfRows, NULL);
      pDst->numOfRows = res->numOfRows;
1683
      pDst->numOfQualified = res->numOfQualified;
D
dapan1121 已提交
1684 1685 1686 1687 1688 1689 1690 1691 1692 1693 1694
    }

    sclFreeParam(res);
    taosHashRemove(ctx.pRes, (void *)&pNode, POINTER_BYTES);
  }

_return:
  sclFreeRes(ctx.pRes);
  return code;
}

H
Hongze Cheng 已提交
1695 1696 1697
int32_t scalarGetOperatorResultType(SOperatorNode *pOp) {
  if (TSDB_DATA_TYPE_BLOB == ((SExprNode *)(pOp->pLeft))->resType.type ||
      (NULL != pOp->pRight && TSDB_DATA_TYPE_BLOB == ((SExprNode *)(pOp->pRight))->resType.type)) {
1698 1699 1700 1701 1702
    return TSDB_CODE_TSC_INVALID_OPERATION;
  }

  switch (pOp->opType) {
    case OP_TYPE_ADD:
D
dapan1121 已提交
1703 1704 1705 1706
    case OP_TYPE_SUB:
    case OP_TYPE_MULTI:
    case OP_TYPE_DIV:
    case OP_TYPE_REM:
D
dapan1121 已提交
1707
      return sclGetMathOperatorResType(pOp);
D
dapan1121 已提交
1708
    case OP_TYPE_MINUS:
D
dapan1121 已提交
1709
      return sclGetMinusOperatorResType(pOp);
1710
    case OP_TYPE_ASSIGN:
H
Hongze Cheng 已提交
1711
      pOp->node.resType = ((SExprNode *)(pOp->pLeft))->resType;
1712 1713 1714
      break;
    case OP_TYPE_BIT_AND:
    case OP_TYPE_BIT_OR:
D
dapan1121 已提交
1715
      return sclGetBitwiseOperatorResType(pOp);
D
dapan1121 已提交
1716 1717 1718 1719 1720 1721
    case OP_TYPE_GREATER_THAN:
    case OP_TYPE_GREATER_EQUAL:
    case OP_TYPE_LOWER_THAN:
    case OP_TYPE_LOWER_EQUAL:
    case OP_TYPE_EQUAL:
    case OP_TYPE_NOT_EQUAL:
1722 1723 1724 1725 1726 1727 1728 1729
    case OP_TYPE_IS_NULL:
    case OP_TYPE_IS_NOT_NULL:
    case OP_TYPE_IS_TRUE:
    case OP_TYPE_IS_FALSE:
    case OP_TYPE_IS_UNKNOWN:
    case OP_TYPE_IS_NOT_TRUE:
    case OP_TYPE_IS_NOT_FALSE:
    case OP_TYPE_IS_NOT_UNKNOWN:
D
dapan1121 已提交
1730 1731 1732 1733
    case OP_TYPE_LIKE:
    case OP_TYPE_NOT_LIKE:
    case OP_TYPE_MATCH:
    case OP_TYPE_NMATCH:
1734 1735
    case OP_TYPE_IN:
    case OP_TYPE_NOT_IN:
D
dapan1121 已提交
1736
      return sclGetCompOperatorResType(pOp);
D
dapan1121 已提交
1737
    case OP_TYPE_JSON_GET_VALUE:
1738
    case OP_TYPE_JSON_CONTAINS:
D
dapan1121 已提交
1739
      return sclGetJsonOperatorResType(pOp);
D
dapan1121 已提交
1740
    default:
1741
      break;
D
dapan1121 已提交
1742 1743
  }

1744 1745
  return TSDB_CODE_SUCCESS;
}