sortTests.cpp 7.7 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27
/*
 * 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/>.
 */

#include <gtest/gtest.h>
#include <tglobal.h>
#include <tsort.h>
#include <iostream>

#pragma GCC diagnostic push
#pragma GCC diagnostic ignored "-Wwrite-strings"
#pragma GCC diagnostic ignored "-Wunused-function"
#pragma GCC diagnostic ignored "-Wunused-variable"
#pragma GCC diagnostic ignored "-Wsign-compare"
#include "os.h"

28
#include "executorimpl.h"
29 30 31
#include "executor.h"
#include "stub.h"
#include "taos.h"
H
Haojun Liao 已提交
32
#include "tdatablock.h"
33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49
#include "tdef.h"
#include "trpc.h"
#include "tvariant.h"

namespace {
typedef struct {
  int32_t startVal;
  int32_t count;
  int32_t pageRows;
} _info;

SSDataBlock* getSingleColDummyBlock(void* param) {
  _info* pInfo = (_info*) param;
  if (--pInfo->count < 0) {
    return NULL;
  }

wafwerar's avatar
wafwerar 已提交
50
  SSDataBlock* pBlock = static_cast<SSDataBlock*>(taosMemoryCalloc(1, sizeof(SSDataBlock)));
51 52 53 54 55 56
  pBlock->pDataBlock = taosArrayInit(4, sizeof(SColumnInfoData));

  SColumnInfoData colInfo = {0};
  colInfo.info.type = TSDB_DATA_TYPE_INT;
  colInfo.info.bytes = sizeof(int32_t);
  colInfo.info.colId = 1;
wafwerar's avatar
wafwerar 已提交
57 58
  colInfo.pData = static_cast<char*>(taosMemoryCalloc(pInfo->pageRows, sizeof(int32_t)));
  colInfo.nullbitmap = static_cast<char*>(taosMemoryCalloc(1, (pInfo->pageRows + 7) / 8));
59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78

  taosArrayPush(pBlock->pDataBlock, &colInfo);

  for (int32_t i = 0; i < pInfo->pageRows; ++i) {
    SColumnInfoData* pColInfo = static_cast<SColumnInfoData*>(TARRAY_GET_ELEM(pBlock->pDataBlock, 0));

    int32_t v = ++pInfo->startVal;
    colDataAppend(pColInfo, i, reinterpret_cast<const char*>(&v), false);
  }

  pBlock->info.rows = pInfo->pageRows;
  pBlock->info.numOfCols = 1;
  return pBlock;
}

int32_t docomp(const void* p1, const void* p2, void* param) {
  int32_t pLeftIdx  = *(int32_t *)p1;
  int32_t pRightIdx = *(int32_t *)p2;

  SMsortComparParam *pParam = (SMsortComparParam *)param;
H
Haojun Liao 已提交
79
  SGenericSource** px = reinterpret_cast<SGenericSource**>(pParam->pSources);
80 81 82

  SArray *pInfo = pParam->orderInfo;

H
Haojun Liao 已提交
83 84
  SGenericSource* pLeftSource  = px[pLeftIdx];
  SGenericSource* pRightSource = px[pRightIdx];
85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100

  // this input is exhausted, set the special value to denote this
  if (pLeftSource->src.rowIndex == -1) {
    return 1;
  }

  if (pRightSource->src.rowIndex == -1) {
    return -1;
  }

  SSDataBlock* pLeftBlock = pLeftSource->src.pBlock;
  SSDataBlock* pRightBlock = pRightSource->src.pBlock;

  for(int32_t i = 0; i < pInfo->size; ++i) {
    SBlockOrderInfo* pOrder = (SBlockOrderInfo*)TARRAY_GET_ELEM(pInfo, i);

H
Haojun Liao 已提交
101
    SColumnInfoData* pLeftColInfoData = (SColumnInfoData*)TARRAY_GET_ELEM(pLeftBlock->pDataBlock, pOrder->slotId);
102 103 104 105 106 107

    bool leftNull  = false;
    if (pLeftColInfoData->hasNull) {
      leftNull = colDataIsNull(pLeftColInfoData, pLeftBlock->info.rows, pLeftSource->src.rowIndex, pLeftBlock->pBlockAgg);
    }

H
Haojun Liao 已提交
108
    SColumnInfoData* pRightColInfoData = (SColumnInfoData*) TARRAY_GET_ELEM(pRightBlock->pDataBlock, pOrder->slotId);
109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125
    bool rightNull = false;
    if (pRightColInfoData->hasNull) {
      rightNull = colDataIsNull(pRightColInfoData, pRightBlock->info.rows, pRightSource->src.rowIndex, pRightBlock->pBlockAgg);
    }

    if (leftNull && rightNull) {
      continue; // continue to next slot
    }

    if (rightNull) {
      return pParam->nullFirst? 1:-1;
    }

    if (leftNull) {
      return pParam->nullFirst? -1:1;
    }

126 127
    void* left1  = colDataGetData(pLeftColInfoData, pLeftSource->src.rowIndex);
    void* right1 = colDataGetData(pRightColInfoData, pRightSource->src.rowIndex);
128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147

    switch(pLeftColInfoData->info.type) {
      case TSDB_DATA_TYPE_INT: {
        int32_t leftv = *(int32_t*)left1;
        int32_t rightv = *(int32_t*)right1;

        if (leftv == rightv) {
          break;
        } else {
          if (pOrder->order == TSDB_ORDER_ASC) {
            return leftv < rightv? -1 : 1;
          } else {
            return leftv < rightv? 1 : -1;
          }
        }
      }
      default:
        assert(0);
    }
  }
H
Haojun Liao 已提交
148 149

  return 0;
150 151 152
}
}  // namespace

wmmhello's avatar
wmmhello 已提交
153
#if 1
154
TEST(testCase, inMem_sort_Test) {
H
Haojun Liao 已提交
155 156
  SBlockOrderInfo oi = {0};
  oi.order = TSDB_ORDER_ASC;
157
  oi.slotId = 0;
H
Haojun Liao 已提交
158 159 160 161
  SArray* orderInfo = taosArrayInit(1, sizeof(SBlockOrderInfo));
  taosArrayPush(orderInfo, &oi);

  SSchema s = {.type = TSDB_DATA_TYPE_INT, .colId = 1, .bytes = 4, };
162
  SSortHandle* phandle = tsortCreateSortHandle(orderInfo, SORT_SINGLESOURCE_SORT, 1024, 5, NULL, "test_abc");
H
Haojun Liao 已提交
163
  tsortSetFetchRawDataFp(phandle, getSingleColDummyBlock);
164 165 166 167 168 169 170 171 172

  _info* pInfo = (_info*) taosMemoryCalloc(1, sizeof(_info));
  pInfo->startVal = 0;
  pInfo->pageRows = 100;
  pInfo->count = 6;

  SGenericSource* ps = static_cast<SGenericSource*>(taosMemoryCalloc(1, sizeof(SGenericSource)));
  ps->param = pInfo;
  tsortAddSource(phandle, ps);
H
Haojun Liao 已提交
173

H
Haojun Liao 已提交
174
  int32_t code = tsortOpen(phandle);
175 176 177
  int32_t row = 1;

  while(1) {
H
Haojun Liao 已提交
178
    STupleHandle* pTupleHandle = tsortNextTuple(phandle);
179 180 181 182
    if (pTupleHandle == NULL) {
      break;
    }

H
Haojun Liao 已提交
183
    void* v = tsortGetValue(pTupleHandle, 0);
184 185
    printf("%d: %d\n", row, *(int32_t*) v);
    ASSERT_EQ(row++, *(int32_t*) v);
186 187

  }
H
Haojun Liao 已提交
188
  tsortDestroySortHandle(phandle);
189 190 191 192 193
}

TEST(testCase, external_mem_sort_Test) {
  SBlockOrderInfo oi = {0};
  oi.order = TSDB_ORDER_ASC;
wmmhello's avatar
wmmhello 已提交
194
  oi.slotId = 0;
195 196 197
  SArray* orderInfo = taosArrayInit(1, sizeof(SBlockOrderInfo));
  taosArrayPush(orderInfo, &oi);

198
  SSortHandle* phandle = tsortCreateSortHandle(orderInfo, SORT_SINGLESOURCE_SORT, 32, 6, NULL, "test_abc");
H
Haojun Liao 已提交
199
  tsortSetFetchRawDataFp(phandle, getSingleColDummyBlock);
H
Haojun Liao 已提交
200

wafwerar's avatar
wafwerar 已提交
201
  _info* pInfo = (_info*) taosMemoryCalloc(1, sizeof(_info));
202 203 204
  pInfo->startVal = 0;
  pInfo->pageRows = 100;
  pInfo->count = 6;
H
Haojun Liao 已提交
205

wafwerar's avatar
wafwerar 已提交
206
  SGenericSource* ps = static_cast<SGenericSource*>(taosMemoryCalloc(1, sizeof(SGenericSource)));
H
Haojun Liao 已提交
207 208
  ps->param = pInfo;

H
Haojun Liao 已提交
209
  tsortAddSource(phandle, ps);
H
Haojun Liao 已提交
210

H
Haojun Liao 已提交
211
  int32_t code = tsortOpen(phandle);
H
Haojun Liao 已提交
212 213 214
  int32_t row = 1;

  while(1) {
H
Haojun Liao 已提交
215
    STupleHandle* pTupleHandle = tsortNextTuple(phandle);
H
Haojun Liao 已提交
216 217 218 219
    if (pTupleHandle == NULL) {
      break;
    }

H
Haojun Liao 已提交
220
    void* v = tsortGetValue(pTupleHandle, 0);
221 222
    printf("%d: %d\n", row, *(int32_t*) v);
    ASSERT_EQ(row++, *(int32_t*) v);
H
Haojun Liao 已提交
223 224

  }
H
Haojun Liao 已提交
225
  tsortDestroySortHandle(phandle);
H
Haojun Liao 已提交
226
}
227

228 229 230
TEST(testCase, ordered_merge_sort_Test) {
  SBlockOrderInfo oi = {0};
  oi.order = TSDB_ORDER_ASC;
231
  oi.slotId = 0;
232 233 234 235
  SArray* orderInfo = taosArrayInit(1, sizeof(SBlockOrderInfo));
  taosArrayPush(orderInfo, &oi);

  SSchema s = {.type = TSDB_DATA_TYPE_INT, .colId = 1, .bytes = 4};
236
  SSortHandle* phandle = tsortCreateSortHandle(orderInfo, SORT_MULTISOURCE_MERGE, 1024, 5, NULL,"test_abc");
H
Haojun Liao 已提交
237 238
  tsortSetFetchRawDataFp(phandle, getSingleColDummyBlock);
  tsortSetComparFp(phandle, docomp);
239 240

  for(int32_t i = 0; i < 10; ++i) {
wafwerar's avatar
wafwerar 已提交
241 242
    SGenericSource* p = static_cast<SGenericSource*>(taosMemoryCalloc(1, sizeof(SGenericSource)));
    _info* c = static_cast<_info*>(taosMemoryCalloc(1, sizeof(_info)));
243 244
    c->count    = 1;
    c->pageRows = 1000;
245
    c->startVal = i*1000;
246 247

    p->param = c;
H
Haojun Liao 已提交
248
    tsortAddSource(phandle, p);
249 250
  }

H
Haojun Liao 已提交
251
  int32_t code = tsortOpen(phandle);
252 253 254
  int32_t row = 1;

  while(1) {
H
Haojun Liao 已提交
255
    STupleHandle* pTupleHandle = tsortNextTuple(phandle);
256 257 258 259
    if (pTupleHandle == NULL) {
      break;
    }

H
Haojun Liao 已提交
260
    void* v = tsortGetValue(pTupleHandle, 0);
261 262
    printf("%d: %d\n", row, *(int32_t*) v);
    ASSERT_EQ(row++, *(int32_t*) v);
263 264

  }
H
Haojun Liao 已提交
265
  tsortDestroySortHandle(phandle);
266 267 268
}

#endif
269 270

#pragma GCC diagnostic pop