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 466
  }

D
dapan1121 已提交
467
  if (pParamList) {
H
Hongze Cheng 已提交
468
    SNode  *tnode = NULL;
D
dapan1121 已提交
469 470
    int32_t i = 0;
    if (SCL_IS_CONST_CALC(ctx)) {
H
Hongze Cheng 已提交
471
      WHERE_EACH(tnode, pParamList) {
D
dapan1121 已提交
472
        if (!SCL_IS_CONST_NODE(tnode)) {
D
dapan 已提交
473
          WHERE_NEXT;
D
dapan1121 已提交
474 475 476 477
        } else {
          SCL_ERR_JRET(sclInitParam(tnode, &paramList[i], ctx, rowNum));
          ERASE_NODE(pParamList);
        }
G
Ganlin Zhao 已提交
478

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

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

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

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

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

wafwerar's avatar
wafwerar 已提交
508
  switch ((int)nodeType(pNode)) {
D
dapan1121 已提交
509 510 511 512 513 514
    case QUERY_NODE_VALUE: {
      SValueNode *valueNode = (SValueNode *)pNode;
      return valueNode->node.resType.type;
    }
    case QUERY_NODE_NODE_LIST: {
      SNodeListNode *nodeList = (SNodeListNode *)pNode;
515
      return nodeList->node.resType.type;
D
dapan1121 已提交
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 541
    }
    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 已提交
542 543
int32_t sclInitOperatorParams(SScalarParam **pParams, SOperatorNode *node, SScalarCtx *ctx, int32_t *rowNum) {
  int32_t code = 0;
D
dapan1121 已提交
544
  int32_t paramNum = scalarGetOperatorParamNum(node->opType);
D
dapan1121 已提交
545 546 547 548
  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 已提交
549

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

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

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

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

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

X
Xiaoyu Wang 已提交
572
int32_t sclGetNodeRes(SNode *node, SScalarCtx *ctx, SScalarParam **res) {
D
dapan1121 已提交
573 574 575 576 577 578 579 580 581
  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 已提交
582

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

  return TSDB_CODE_SUCCESS;
}

X
Xiaoyu Wang 已提交
588 589 590 591 592 593 594 595 596 597 598 599
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 已提交
600 601 602

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

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

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

      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 已提交
616

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

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

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

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

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

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

_return:
X
Xiaoyu Wang 已提交
642

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

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

X
Xiaoyu Wang 已提交
651 652 653 654 655 656 657
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 已提交
658

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

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

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

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

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

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

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

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

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

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

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

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

D
dapan1121 已提交
709 710 711 712
_return:

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

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

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

D
dapan1121 已提交
726
  if (fmIsUserDefinedFunc(node->funcId)) {
727
    code = callUdfScalarFunc(node->functionName, params, paramNum, output);
728 729 730 731
    if (code != 0) {
      sclError("fmExecFunction error. callUdfScalarFunc. function name: %s, udf code:%d", node->functionName, code);
      goto _return;
    }
D
dapan1121 已提交
732 733 734 735 736 737 738
  } 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 已提交
739

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

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

D
dapan1121 已提交
749 750 751 752 753
    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 已提交
754 755 756 757
  }

_return:

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

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

D
dapan1121 已提交
774 775 776 777 778 779
  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 已提交
780 781 782
  int32_t       rowNum = 0;
  int32_t       paramNum = 0;
  int32_t       code = 0;
D
dapan 已提交
783
  SCL_ERR_RET(sclInitParamList(&params, node->pParameterList, ctx, &paramNum, &rowNum));
D
dapan1121 已提交
784 785 786 787
  if (NULL == params) {
    output->numOfRows = 0;
    return TSDB_CODE_SUCCESS;
  }
D
dapan1121 已提交
788

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

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

798 799
  int32_t numOfQualified = 0;

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

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

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

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

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

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

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

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

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

  _bin_scalar_fn_t OperatorFn = getBinScalarOperatorFn(node->opType);

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

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

_return:
D
dapan1121 已提交
876

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

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

  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 已提交
897
    SSDataBlock *pb = taosArrayGetP(ctx->pBlockList, 0);
D
dapan1121 已提交
898
    rowNum = pb->info.rows;
D
dapan1121 已提交
899
    output->numOfRows = pb->info.rows;
D
dapan1121 已提交
900
  }
D
dapan1121 已提交
901 902

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

D
dapan1121 已提交
904 905 906 907 908 909
  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 已提交
910

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

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

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

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

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

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

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

_return:

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

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

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

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

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

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

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

  return DEAL_RES_CONTINUE;
}

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

  return DEAL_RES_CONTINUE;
}

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

1041 1042 1043 1044

int32_t sclConvertOpValueNodeTs(SOperatorNode *node, SScalarCtx *ctx) {
  int32_t code = 0;
  
D
dapan1121 已提交
1045
  if (node->pLeft && SCL_IS_VAR_VALUE_NODE(node->pLeft)) {
1046 1047 1048
    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 已提交
1049
  } else if (node->pRight && SCL_IS_NOTNULL_CONST_NODE(node->pRight)) {
1050 1051 1052
    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 已提交
1053
      } else if (QUERY_NODE_NODE_LIST == node->pRight->type) {
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 1106
        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 已提交
1107
EDealRes sclRewriteNonConstOperator(SNode **pNode, SScalarCtx *ctx) {
D
dapan1121 已提交
1108
  SOperatorNode *node = (SOperatorNode *)*pNode;
H
Hongze Cheng 已提交
1109
  int32_t        code = 0;
D
dapan1121 已提交
1110 1111 1112

  if (node->pLeft && (QUERY_NODE_VALUE == nodeType(node->pLeft))) {
    SValueNode *valueNode = (SValueNode *)node->pLeft;
H
Hongze Cheng 已提交
1113 1114
    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 已提交
1115
      return sclRewriteNullInOptr(pNode, ctx, node->opType);
D
dapan1121 已提交
1116
    }
D
dapan 已提交
1117

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

  if (node->pRight && (QUERY_NODE_VALUE == nodeType(node->pRight))) {
    SValueNode *valueNode = (SValueNode *)node->pRight;
H
Hongze Cheng 已提交
1125 1126
    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 已提交
1127
      return sclRewriteNullInOptr(pNode, ctx, node->opType);
D
dapan1121 已提交
1128
    }
D
dapan 已提交
1129

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

  if (node->pRight && (QUERY_NODE_NODE_LIST == nodeType(node->pRight))) {
    SNodeListNode *listNode = (SNodeListNode *)node->pRight;
H
Hongze Cheng 已提交
1137
    SNode         *tnode = NULL;
D
dapan1121 已提交
1138 1139 1140 1141 1142
    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 已提交
1143
        } else {  // OP_TYPE_NOT_IN
D
dapan1121 已提交
1144
          return sclRewriteNullInOptr(pNode, ctx, node->opType);
D
dapan1121 已提交
1145 1146 1147 1148 1149 1150 1151
        }
      }

      WHERE_NEXT;
    }

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

  return DEAL_RES_CONTINUE;
}

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

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

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

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

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

1187 1188
  res->translate = true;

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

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

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

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

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

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

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

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

1242 1243 1244 1245
  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 已提交
1246
  } else {
D
dapan1121 已提交
1247
    nodesSetValueNodeValue(res, output.columnData->pData);
D
dapan1121 已提交
1248
  }
D
dapan1121 已提交
1249 1250

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

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

1257

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

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

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

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

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

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

1284
  res->node.resType = node->node.resType;
1285
  if (colDataIsNull_s(output.columnData, 0)) {
1286 1287
    res->isNull = true;
    res->node.resType = node->node.resType;
1288
  } else {
1289 1290 1291 1292 1293
    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 已提交
1294
      nodesSetValueNodeValue(res, output.columnData->pData);
1295
    }
D
dapan1121 已提交
1296
  }
D
dapan1121 已提交
1297 1298

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

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

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

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

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

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

  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 已提交
1332
    ctx->code = TSDB_CODE_OUT_OF_MEMORY;
D
dapan1121 已提交
1333 1334 1335 1336 1337 1338 1339 1340 1341 1342 1343 1344 1345 1346 1347 1348 1349 1350 1351 1352
    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 已提交
1353
  *pNode = (SNode *)res;
D
dapan1121 已提交
1354 1355 1356 1357 1358

  sclFreeParam(&output);
  return DEAL_RES_CONTINUE;
}

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

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

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

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

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

  return DEAL_RES_CONTINUE;
D
dapan1121 已提交
1379 1380
}

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

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

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

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

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

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

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

  return DEAL_RES_CONTINUE;
}

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

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

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

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

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

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

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

  if (index == -1) {
H
Hongze Cheng 已提交
1453 1454
    sclError("column tupleId is too big, tupleId:%d, dataBlockNum:%d", target->dataBlockId,
             (int32_t)taosArrayGetSize(ctx->pBlockList));
1455 1456 1457 1458 1459 1460 1461
    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 已提交
1462 1463
    sclError("target slot not exist, dataBlockId:%d, slotId:%d, dataBlockNum:%d", target->dataBlockId, target->slotId,
             (int32_t)taosArrayGetSize(block->pDataBlock));
1464 1465 1466 1467 1468 1469 1470 1471 1472 1473
    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 已提交
1474
    ctx->code = TSDB_CODE_APP_ERROR;
1475 1476 1477
    return DEAL_RES_ERROR;
  }

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

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

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

  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 已提交
1496
    ctx->code = TSDB_CODE_OUT_OF_MEMORY;
D
dapan1121 已提交
1497 1498 1499 1500 1501 1502
    return DEAL_RES_ERROR;
  }

  return DEAL_RES_CONTINUE;
}

H
Hongze Cheng 已提交
1503
EDealRes sclCalcWalker(SNode *pNode, void *pContext) {
X
Xiaoyu Wang 已提交
1504 1505 1506
  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 已提交
1507
    return DEAL_RES_CONTINUE;
D
dapan1121 已提交
1508
  }
D
dapan 已提交
1509 1510

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

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

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

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

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

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

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

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

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

D
dapan 已提交
1554
_return:
1555

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

H
Hongze Cheng 已提交
1560 1561
static int32_t sclGetMinusOperatorResType(SOperatorNode *pOp) {
  if (!IS_MATHABLE_TYPE(((SExprNode *)(pOp->pLeft))->resType.type)) {
1562 1563 1564 1565 1566 1567 1568
    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 已提交
1569
static int32_t sclGetMathOperatorResType(SOperatorNode *pOp) {
G
Ganlin Zhao 已提交
1570 1571 1572 1573
  if (pOp == NULL || pOp->pLeft == NULL || pOp->pRight == NULL) {
    return TSDB_CODE_TSC_INVALID_OPERATION;
  }

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

1577 1578 1579 1580 1581 1582 1583 1584 1585 1586 1587 1588 1589 1590 1591 1592 1593 1594 1595
  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 已提交
1596
static int32_t sclGetCompOperatorResType(SOperatorNode *pOp) {
G
Ganlin Zhao 已提交
1597
  if (pOp == NULL || pOp->pLeft == NULL) {
G
Ganlin Zhao 已提交
1598 1599 1600
    return TSDB_CODE_TSC_INVALID_OPERATION;
  }

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

1603
  if (OP_TYPE_IN == pOp->opType || OP_TYPE_NOT_IN == pOp->opType) {
G
Ganlin Zhao 已提交
1604 1605 1606
    if (pOp->pRight == NULL) {
      return TSDB_CODE_TSC_INVALID_OPERATION;
    }
H
Hongze Cheng 已提交
1607
    ((SExprNode *)(pOp->pRight))->resType = ldt;
1608
  } else if (nodesIsRegularOp(pOp)) {
G
Ganlin Zhao 已提交
1609 1610 1611
    if (pOp->pRight == NULL) {
      return TSDB_CODE_TSC_INVALID_OPERATION;
    }
H
Hongze Cheng 已提交
1612
    SDataType rdt = ((SExprNode *)(pOp->pRight))->resType;
1613 1614 1615
    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;
1616 1617 1618 1619 1620 1621 1622
    }
  }
  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 已提交
1623
static int32_t sclGetJsonOperatorResType(SOperatorNode *pOp) {
G
Ganlin Zhao 已提交
1624 1625 1626 1627
  if (pOp == NULL || pOp->pLeft == NULL || pOp->pRight == NULL) {
    return TSDB_CODE_TSC_INVALID_OPERATION;
  }

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

1631 1632 1633 1634 1635 1636 1637 1638 1639 1640 1641 1642
  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 已提交
1643
static int32_t sclGetBitwiseOperatorResType(SOperatorNode *pOp) {
1644 1645 1646 1647 1648
  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 已提交
1649
int32_t scalarCalculateConstants(SNode *pNode, SNode **pRes) { return sclCalcConstants(pNode, false, pRes); }
D
dapan1121 已提交
1650

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

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 已提交
1658
  int32_t    code = 0;
D
dapan1121 已提交
1659 1660 1661 1662 1663 1664
  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 已提交
1665
    SCL_ERR_RET(TSDB_CODE_OUT_OF_MEMORY);
D
dapan1121 已提交
1666
  }
1667

D
dapan1121 已提交
1668 1669 1670 1671 1672 1673 1674
  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 已提交
1675
      SCL_ERR_JRET(TSDB_CODE_APP_ERROR);
D
dapan1121 已提交
1676 1677 1678 1679 1680
    }

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

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

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

H
Hongze Cheng 已提交
1696 1697 1698
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)) {
1699 1700 1701 1702 1703
    return TSDB_CODE_TSC_INVALID_OPERATION;
  }

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

1745 1746
  return TSDB_CODE_SUCCESS;
}