sortTests.cpp 7.5 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
#include "tdef.h"
#include "trpc.h"
#include "tvariant.h"
36
#include "tcompare.h"
37 38 39 40 41 42 43 44 45 46 47 48 49 50

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 已提交
51
  SSDataBlock* pBlock = static_cast<SSDataBlock*>(taosMemoryCalloc(1, sizeof(SSDataBlock)));
52 53 54 55 56 57
  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 已提交
58 59
  colInfo.pData = static_cast<char*>(taosMemoryCalloc(pInfo->pageRows, sizeof(int32_t)));
  colInfo.nullbitmap = static_cast<char*>(taosMemoryCalloc(1, (pInfo->pageRows + 7) / 8));
60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79

  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 已提交
80
  SGenericSource** px = reinterpret_cast<SGenericSource**>(pParam->pSources);
81 82 83

  SArray *pInfo = pParam->orderInfo;

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

  // 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 已提交
102
    SColumnInfoData* pLeftColInfoData = (SColumnInfoData*)TARRAY_GET_ELEM(pLeftBlock->pDataBlock, pOrder->slotId);
103 104 105 106 107 108

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

H
Haojun Liao 已提交
109
    SColumnInfoData* pRightColInfoData = (SColumnInfoData*) TARRAY_GET_ELEM(pRightBlock->pDataBlock, pOrder->slotId);
110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126
    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;
    }

127 128
    void* left1  = colDataGetData(pLeftColInfoData, pLeftSource->src.rowIndex);
    void* right1 = colDataGetData(pRightColInfoData, pRightSource->src.rowIndex);
129
    __compar_fn_t fn = getKeyComparFunc(pLeftColInfoData->info.type, pOrder->order);
130

131 132 133 134 135
    int ret = fn(left1, right1);
    if (ret == 0) {
      continue;
    } else {
      return ret;
136 137
    }
  }
H
Haojun Liao 已提交
138 139

  return 0;
140 141 142
}
}  // namespace

wmmhello's avatar
wmmhello 已提交
143
#if 1
144
TEST(testCase, inMem_sort_Test) {
H
Haojun Liao 已提交
145 146
  SBlockOrderInfo oi = {0};
  oi.order = TSDB_ORDER_ASC;
147
  oi.slotId = 0;
H
Haojun Liao 已提交
148 149 150 151
  SArray* orderInfo = taosArrayInit(1, sizeof(SBlockOrderInfo));
  taosArrayPush(orderInfo, &oi);

  SSchema s = {.type = TSDB_DATA_TYPE_INT, .colId = 1, .bytes = 4, };
152
  SSortHandle* phandle = tsortCreateSortHandle(orderInfo, SORT_SINGLESOURCE_SORT, 1024, 5, NULL, "test_abc");
H
Haojun Liao 已提交
153
  tsortSetFetchRawDataFp(phandle, getSingleColDummyBlock);
154 155 156 157 158 159 160 161 162

  _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 已提交
163

H
Haojun Liao 已提交
164
  int32_t code = tsortOpen(phandle);
165 166 167
  int32_t row = 1;

  while(1) {
H
Haojun Liao 已提交
168
    STupleHandle* pTupleHandle = tsortNextTuple(phandle);
169 170 171 172
    if (pTupleHandle == NULL) {
      break;
    }

H
Haojun Liao 已提交
173
    void* v = tsortGetValue(pTupleHandle, 0);
174 175
    printf("%d: %d\n", row, *(int32_t*) v);
    ASSERT_EQ(row++, *(int32_t*) v);
176 177

  }
H
Haojun Liao 已提交
178
  tsortDestroySortHandle(phandle);
179 180 181 182 183
}

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

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

wafwerar's avatar
wafwerar 已提交
191
  _info* pInfo = (_info*) taosMemoryCalloc(1, sizeof(_info));
192 193 194
  pInfo->startVal = 0;
  pInfo->pageRows = 100;
  pInfo->count = 6;
H
Haojun Liao 已提交
195

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

H
Haojun Liao 已提交
199
  tsortAddSource(phandle, ps);
H
Haojun Liao 已提交
200

H
Haojun Liao 已提交
201
  int32_t code = tsortOpen(phandle);
H
Haojun Liao 已提交
202 203 204
  int32_t row = 1;

  while(1) {
H
Haojun Liao 已提交
205
    STupleHandle* pTupleHandle = tsortNextTuple(phandle);
H
Haojun Liao 已提交
206 207 208 209
    if (pTupleHandle == NULL) {
      break;
    }

H
Haojun Liao 已提交
210
    void* v = tsortGetValue(pTupleHandle, 0);
211 212
    printf("%d: %d\n", row, *(int32_t*) v);
    ASSERT_EQ(row++, *(int32_t*) v);
H
Haojun Liao 已提交
213 214

  }
H
Haojun Liao 已提交
215
  tsortDestroySortHandle(phandle);
H
Haojun Liao 已提交
216
}
217

218 219 220
TEST(testCase, ordered_merge_sort_Test) {
  SBlockOrderInfo oi = {0};
  oi.order = TSDB_ORDER_ASC;
221
  oi.slotId = 0;
222 223 224 225
  SArray* orderInfo = taosArrayInit(1, sizeof(SBlockOrderInfo));
  taosArrayPush(orderInfo, &oi);

  SSchema s = {.type = TSDB_DATA_TYPE_INT, .colId = 1, .bytes = 4};
226
  SSortHandle* phandle = tsortCreateSortHandle(orderInfo, SORT_MULTISOURCE_MERGE, 1024, 5, NULL,"test_abc");
H
Haojun Liao 已提交
227 228
  tsortSetFetchRawDataFp(phandle, getSingleColDummyBlock);
  tsortSetComparFp(phandle, docomp);
229 230

  for(int32_t i = 0; i < 10; ++i) {
wafwerar's avatar
wafwerar 已提交
231 232
    SGenericSource* p = static_cast<SGenericSource*>(taosMemoryCalloc(1, sizeof(SGenericSource)));
    _info* c = static_cast<_info*>(taosMemoryCalloc(1, sizeof(_info)));
233 234
    c->count    = 1;
    c->pageRows = 1000;
235
    c->startVal = i*1000;
236 237

    p->param = c;
H
Haojun Liao 已提交
238
    tsortAddSource(phandle, p);
239 240
  }

H
Haojun Liao 已提交
241
  int32_t code = tsortOpen(phandle);
242 243 244
  int32_t row = 1;

  while(1) {
H
Haojun Liao 已提交
245
    STupleHandle* pTupleHandle = tsortNextTuple(phandle);
246 247 248 249
    if (pTupleHandle == NULL) {
      break;
    }

H
Haojun Liao 已提交
250
    void* v = tsortGetValue(pTupleHandle, 0);
251 252
    printf("%d: %d\n", row, *(int32_t*) v);
    ASSERT_EQ(row++, *(int32_t*) v);
253 254

  }
H
Haojun Liao 已提交
255
  tsortDestroySortHandle(phandle);
256 257 258
}

#endif
259 260

#pragma GCC diagnostic pop