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

16
#include "function.h"
S
slguan 已提交
17
#include "os.h"
H
Haojun Liao 已提交
18

S
Shengliang Guan 已提交
19
#include "texception.h"
20
#include "taosdef.h"
H
Hongze Cheng 已提交
21
#include "tmsg.h"
H
Haojun Liao 已提交
22 23 24
#include "tarray.h"
#include "tbuffer.h"
#include "tcompare.h"
H
Haojun Liao 已提交
25
#include "thash.h"
H
Haojun Liao 已提交
26
#include "texpr.h"
H
Haojun Liao 已提交
27
#include "tvariant.h"
D
dapan1121 已提交
28
#include "tdef.h"
H
hzcheng 已提交
29

H
Haojun Liao 已提交
30 31 32
static void doExprTreeDestroy(tExprNode **pExpr, void (*fp)(void *));

void tExprTreeDestroy(tExprNode *pNode, void (*fp)(void *)) {
H
Haojun Liao 已提交
33 34 35 36
  if (pNode == NULL) {
    return;
  }

37
  if (pNode->nodeType == TEXPR_BINARYEXPR_NODE || pNode->nodeType == TEXPR_UNARYEXPR_NODE) {
H
Haojun Liao 已提交
38
    doExprTreeDestroy(&pNode, fp);
H
Haojun Liao 已提交
39
  }
wafwerar's avatar
wafwerar 已提交
40
  taosMemoryFree(pNode);
H
Haojun Liao 已提交
41 42
}

H
Haojun Liao 已提交
43
static void doExprTreeDestroy(tExprNode **pExpr, void (*fp)(void *)) {
H
Haojun Liao 已提交
44 45 46
  if (*pExpr == NULL) {
    return;
  }
wafwerar's avatar
wafwerar 已提交
47
  taosMemoryFree(*pExpr);
H
Haojun Liao 已提交
48 49 50
  *pExpr = NULL;
}

H
Haojun Liao 已提交
51
bool exprTreeApplyFilter(tExprNode *pExpr, const void *pItem, SExprTraverseSupp *param) {
52
#if 0
H
Haojun Liao 已提交
53
  //non-leaf nodes, recursively traverse the expression tree in the post-root order
54
  if (pLeft->nodeType == TEXPR_BINARYEXPR_NODE && pRight->nodeType == TEXPR_BINARYEXPR_NODE) {
D
dapan1121 已提交
55
    if (pExpr->_node.optr == LOGIC_COND_TYPE_OR) {  // or
H
Haojun Liao 已提交
56
      if (exprTreeApplyFilter(pLeft, pItem, param)) {
H
Haojun Liao 已提交
57 58 59 60
        return true;
      }

      // left child does not satisfy the query condition, try right child
H
Haojun Liao 已提交
61
      return exprTreeApplyFilter(pRight, pItem, param);
H
Haojun Liao 已提交
62
    } else {  // and
H
Haojun Liao 已提交
63
      if (!exprTreeApplyFilter(pLeft, pItem, param)) {
H
Haojun Liao 已提交
64 65 66
        return false;
      }

H
Haojun Liao 已提交
67
      return exprTreeApplyFilter(pRight, pItem, param);
H
Haojun Liao 已提交
68 69 70 71 72 73
    }
  }

  // handle the leaf node
  param->setupInfoFn(pExpr, param->pExtInfo);
  return param->nodeFilterFn(pItem, pExpr->_node.info);
74 75 76
#endif

  return 0;
H
Haojun Liao 已提交
77 78
}

79 80
// TODO: these three functions should be made global
static void* exception_calloc(size_t nmemb, size_t size) {
wafwerar's avatar
wafwerar 已提交
81
  void* p = taosMemoryCalloc(nmemb, size);
82
  if (p == NULL) {
83
    THROW(TSDB_CODE_QRY_OUT_OF_MEMORY);
84
  }
85 86 87 88
  return p;
}

static void* exception_malloc(size_t size) {
wafwerar's avatar
wafwerar 已提交
89
  void* p = taosMemoryMalloc(size);
90
  if (p == NULL) {
91
    THROW(TSDB_CODE_QRY_OUT_OF_MEMORY);
92 93
  }
  return p;
94 95
}

96
static UNUSED_FUNC char* exception_strdup(const char* str) {
97 98
  char* p = strdup(str);
  if (p == NULL) {
99
    THROW(TSDB_CODE_QRY_OUT_OF_MEMORY);
100 101 102 103
  }
  return p;
}

Y
yihaoDeng 已提交
104 105 106 107
void buildFilterSetFromBinary(void **q, const char *buf, int32_t len) {
  SBufferReader br = tbufInitReader(buf, len, false); 
  uint32_t type  = tbufReadUint32(&br);     
  SHashObj *pObj = taosHashInit(256, taosGetDefaultHashFunction(type), true, false);
108
  
H
Haojun Liao 已提交
109 110
//  taosHashSetEqualFp(pObj, taosGetDefaultEqualFunction(type));

Y
yihaoDeng 已提交
111 112 113
  int dummy = -1;
  int32_t sz = tbufReadInt32(&br);
  for (int32_t i = 0; i < sz; i++) {
114
    if (type == TSDB_DATA_TYPE_BOOL || IS_SIGNED_NUMERIC_TYPE(type)) {
115 116
      int64_t val = tbufReadInt64(&br); 
      taosHashPut(pObj, (char *)&val, sizeof(val),  &dummy, sizeof(dummy));
117 118 119 120 121
    } else if (IS_UNSIGNED_NUMERIC_TYPE(type)) {
      uint64_t val = tbufReadUint64(&br); 
      taosHashPut(pObj, (char *)&val, sizeof(val),  &dummy, sizeof(dummy));
    }
    else if (type == TSDB_DATA_TYPE_TIMESTAMP) {
Y
yihaoDeng 已提交
122 123 124 125 126
      int64_t val = tbufReadInt64(&br); 
      taosHashPut(pObj, (char *)&val, sizeof(val),  &dummy, sizeof(dummy));
    } else if (type == TSDB_DATA_TYPE_DOUBLE || type == TSDB_DATA_TYPE_FLOAT) {
      double  val = tbufReadDouble(&br);
      taosHashPut(pObj, (char *)&val, sizeof(val), &dummy, sizeof(dummy));
Y
yihaoDeng 已提交
127
    } else if (type == TSDB_DATA_TYPE_BINARY) {
Y
yihaoDeng 已提交
128 129 130
      size_t  t = 0;
      const char *val = tbufReadBinary(&br, &t);
      taosHashPut(pObj, (char *)val, t, &dummy, sizeof(dummy));
Y
yihaoDeng 已提交
131
    } else if (type == TSDB_DATA_TYPE_NCHAR) {
Y
yihaoDeng 已提交
132 133 134
      size_t  t = 0;
      const char *val = tbufReadBinary(&br, &t);      
      taosHashPut(pObj, (char *)val, t, &dummy, sizeof(dummy));
Y
yihaoDeng 已提交
135 136 137 138 139
    }
  } 
  *q = (void *)pObj;
}

W
wpan 已提交
140 141 142 143 144
void convertFilterSetFromBinary(void **q, const char *buf, int32_t len, uint32_t tType) {
  SBufferReader br = tbufInitReader(buf, len, false); 
  uint32_t sType  = tbufReadUint32(&br);     
  SHashObj *pObj = taosHashInit(256, taosGetDefaultHashFunction(tType), true, false);
  
H
Haojun Liao 已提交
145
//  taosHashSetEqualFp(pObj, taosGetDefaultEqualFunction(tType));
W
wpan 已提交
146 147
  
  int dummy = -1;
H
Haojun Liao 已提交
148
  SVariant tmpVar = {0};  
W
wpan 已提交
149 150
  size_t  t = 0;
  int32_t sz = tbufReadInt32(&br);
W
wpan 已提交
151 152
  void *pvar = NULL;  
  int64_t val = 0;
W
wpan 已提交
153 154 155 156 157 158 159
  int32_t bufLen = 0;
  if (IS_NUMERIC_TYPE(sType)) {
    bufLen = 60;  // The maximum length of string that a number is converted to.
  } else {
    bufLen = 128;
  }

wafwerar's avatar
wafwerar 已提交
160
  char *tmp = taosMemoryCalloc(1, bufLen * TSDB_NCHAR_SIZE);
W
wpan 已提交
161 162 163 164
    
  for (int32_t i = 0; i < sz; i++) {
    switch (sType) {
    case TSDB_DATA_TYPE_BOOL:
W
wpan 已提交
165
    case TSDB_DATA_TYPE_UTINYINT:
W
wpan 已提交
166
    case TSDB_DATA_TYPE_TINYINT: {
W
wpan 已提交
167
      *(uint8_t *)&val = (uint8_t)tbufReadInt64(&br); 
W
wpan 已提交
168 169 170 171
      t = sizeof(val);
      pvar = &val;
      break;
    }
W
wpan 已提交
172
    case TSDB_DATA_TYPE_USMALLINT:
W
wpan 已提交
173
    case TSDB_DATA_TYPE_SMALLINT: {
W
wpan 已提交
174
      *(uint16_t *)&val = (uint16_t)tbufReadInt64(&br); 
W
wpan 已提交
175 176 177 178
      t = sizeof(val);
      pvar = &val;
      break;
    }
W
wpan 已提交
179
    case TSDB_DATA_TYPE_UINT:
W
wpan 已提交
180
    case TSDB_DATA_TYPE_INT: {
W
wpan 已提交
181
      *(uint32_t *)&val = (uint32_t)tbufReadInt64(&br); 
W
wpan 已提交
182 183 184 185
      t = sizeof(val);
      pvar = &val;
      break;
    }
W
wpan 已提交
186
    case TSDB_DATA_TYPE_TIMESTAMP:
W
wpan 已提交
187
    case TSDB_DATA_TYPE_UBIGINT:
W
wpan 已提交
188
    case TSDB_DATA_TYPE_BIGINT: {
W
wpan 已提交
189
      *(uint64_t *)&val = (uint64_t)tbufReadInt64(&br); 
W
wpan 已提交
190 191 192 193 194
      t = sizeof(val);
      pvar = &val;
      break;
    }
    case TSDB_DATA_TYPE_DOUBLE: {
W
wpan 已提交
195
      *(double *)&val = tbufReadDouble(&br);
W
wpan 已提交
196 197 198 199 200
      t = sizeof(val);
      pvar = &val;
      break;
    }
    case TSDB_DATA_TYPE_FLOAT: {
W
wpan 已提交
201
      *(float *)&val = (float)tbufReadDouble(&br);
W
wpan 已提交
202 203 204 205 206
      t = sizeof(val);
      pvar = &val;
      break;
    }
    case TSDB_DATA_TYPE_BINARY: {
W
wpan 已提交
207
      pvar = (char *)tbufReadBinary(&br, &t);
W
wpan 已提交
208 209 210
      break;
    }
    case TSDB_DATA_TYPE_NCHAR: {
W
wpan 已提交
211
      pvar = (char *)tbufReadBinary(&br, &t);      
W
wpan 已提交
212 213 214 215 216 217 218 219
      break;
    }
    default:
      taosHashCleanup(pObj);
      *q = NULL;
      return;
    }
    
H
Haojun Liao 已提交
220
    taosVariantCreateFromBinary(&tmpVar, (char *)pvar, t, sType);
W
wpan 已提交
221 222

    if (bufLen < t) {
wafwerar's avatar
wafwerar 已提交
223
      tmp = taosMemoryRealloc(tmp, t * TSDB_NCHAR_SIZE);
W
wpan 已提交
224
      bufLen = (int32_t)t;
W
wpan 已提交
225 226 227 228
    }

    switch (tType) {
      case TSDB_DATA_TYPE_BOOL:
W
wpan 已提交
229
      case TSDB_DATA_TYPE_UTINYINT:
W
wpan 已提交
230
      case TSDB_DATA_TYPE_TINYINT: {
H
Haojun Liao 已提交
231
        if (taosVariantDump(&tmpVar, (char *)&val, tType, false)) {
W
wpan 已提交
232 233 234 235 236 237
          goto err_ret;
        }
        pvar = &val;
        t = sizeof(val);
        break;
      }
W
wpan 已提交
238
      case TSDB_DATA_TYPE_USMALLINT:
W
wpan 已提交
239
      case TSDB_DATA_TYPE_SMALLINT: {
H
Haojun Liao 已提交
240
        if (taosVariantDump(&tmpVar, (char *)&val, tType, false)) {
W
wpan 已提交
241 242 243 244 245 246
          goto err_ret;
        }
        pvar = &val;
        t = sizeof(val);
        break;
      }
W
wpan 已提交
247
      case TSDB_DATA_TYPE_UINT:
W
wpan 已提交
248
      case TSDB_DATA_TYPE_INT: {
H
Haojun Liao 已提交
249
        if (taosVariantDump(&tmpVar, (char *)&val, tType, false)) {
W
wpan 已提交
250 251 252 253 254 255
          goto err_ret;
        }
        pvar = &val;
        t = sizeof(val);
        break;
      }
W
wpan 已提交
256
      case TSDB_DATA_TYPE_TIMESTAMP:
W
wpan 已提交
257
      case TSDB_DATA_TYPE_UBIGINT:
W
wpan 已提交
258
      case TSDB_DATA_TYPE_BIGINT: {
H
Haojun Liao 已提交
259
        if (taosVariantDump(&tmpVar, (char *)&val, tType, false)) {
W
wpan 已提交
260 261 262 263 264 265 266
          goto err_ret;
        }
        pvar = &val;
        t = sizeof(val);
        break;
      }
      case TSDB_DATA_TYPE_DOUBLE: {
H
Haojun Liao 已提交
267
        if (taosVariantDump(&tmpVar, (char *)&val, tType, false)) {
W
wpan 已提交
268 269 270 271 272 273 274
          goto err_ret;
        }
        pvar = &val;
        t = sizeof(val);
        break;
      }
      case TSDB_DATA_TYPE_FLOAT: {
H
Haojun Liao 已提交
275
        if (taosVariantDump(&tmpVar, (char *)&val, tType, false)) {
W
wpan 已提交
276 277 278 279 280 281 282
          goto err_ret;
        }
        pvar = &val;
        t = sizeof(val);
        break;
      }
      case TSDB_DATA_TYPE_BINARY: {
H
Haojun Liao 已提交
283
        if (taosVariantDump(&tmpVar, tmp, tType, true)) {
W
wpan 已提交
284 285 286 287 288 289 290
          goto err_ret;
        }
        t = varDataLen(tmp);
        pvar = varDataVal(tmp);
        break;
      }
      case TSDB_DATA_TYPE_NCHAR: {
H
Haojun Liao 已提交
291
        if (taosVariantDump(&tmpVar, tmp, tType, true)) {
W
wpan 已提交
292 293 294 295 296 297 298 299 300 301 302
          goto err_ret;
        }
        t = varDataLen(tmp);
        pvar = varDataVal(tmp);        
        break;
      }
      default:
        goto err_ret;
    }
    
    taosHashPut(pObj, (char *)pvar, t,  &dummy, sizeof(dummy));
H
Haojun Liao 已提交
303
    taosVariantDestroy(&tmpVar);
W
wpan 已提交
304
    memset(&tmpVar, 0, sizeof(tmpVar));
W
wpan 已提交
305 306 307 308 309 310
  } 

  *q = (void *)pObj;
  pObj = NULL;
  
err_ret:  
H
Haojun Liao 已提交
311
  taosVariantDestroy(&tmpVar);
W
wpan 已提交
312
  taosHashCleanup(pObj);
wafwerar's avatar
wafwerar 已提交
313
  taosMemoryFreeClear(tmp);
W
wpan 已提交
314
}